2020-09-30 16:53:55 +02:00

6428 lines
209 KiB
C

/******************************************************************************
Copyright(c) Microsoft Corporation
Module Name:
BootCfg64.cpp
Abstract:
This file is intended to have the functionality for
configuring, displaying, changing and deleting boot.ini
settings for the local host for a 64 bit system.
Author:
J.S.Vasu 17/1/2001 .
Revision History:
J.S.Vasu 17/1/2001 Created it.
SanthoshM.B 10/2/2001 Modified it.
J.S.Vasu 15/2/2001 Modified it.
******************************************************************************/
#include "pch.h"
#include "resource.h"
#include "BootCfg.h"
#include "BootCfg64.h"
#include <strsafe.h>
#define BOOTFILE_PATH1 L"signature({%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x})%s"
#define BOOTFILE_PATH L"signature({%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x})%s\\ia64ldr.efi"
NTSTATUS ModifyBootEntry( IN WCHAR *pwszInstallPath, IN PBOOT_ENTRY pSourceEntry);
NTSTATUS FindBootEntry(IN PVOID pEntryListHead,IN WCHAR *pwszTarget, OUT PBOOT_ENTRY *ppTargetEntry);
LPVOID MEMALLOC( ULONG size ) ;
VOID MEMFREE ( LPVOID block ) ;
NTSTATUS EnumerateBootEntries( IN PVOID *ppEntryListHead);
NTSTATUS AcquirePrivilege(IN CONST ULONG ulPrivilege,IN CONST BOOLEAN bEnable);
DWORD ListDeviceInfo(DWORD dwVal);
//Global Linked lists for storing the boot entries
LIST_ENTRY BootEntries;
LIST_ENTRY ActiveUnorderedBootEntries;
LIST_ENTRY InactiveUnorderedBootEntries;
DWORD InitializeEFI(void)
/*++
Routine Description :
This routine initializes the EFI environment required.
Initializes the function pointers for the
NT Boot Entry Management API's
Arguments : None
Return Type : DWORD
Returns EXIT_SUCCESS if successful,
returns EXIT_FAILURE otherwise.
--*/
{
DWORD error;
NTSTATUS status;
BOOLEAN wasEnabled;
HMODULE hModule;
PBOOT_ENTRY_LIST ntBootEntries = NULL;
PMY_BOOT_ENTRY bootEntry;
PLIST_ENTRY listEntry;
PULONG BootEntryOrder;
ULONG BootEntryOrderCount;
ULONG length, i, myId;
TCHAR dllName[MAX_PATH];
if( FALSE == IsUserAdmin() )
{
ShowMessage( stderr, GetResString(IDS_NOT_ADMINISTRATOR_64 ));
ReleaseGlobals();
return EXIT_FAILURE;
}
// Enable the privilege that is necessary to query/set NVRAM.
status = RtlAdjustPrivilege(SE_SYSTEM_ENVIRONMENT_PRIVILEGE,
TRUE,
FALSE,
&wasEnabled
);
if (!NT_SUCCESS(status))
{
error = RtlNtStatusToDosError( status );
ShowMessage( stderr, GetResString(IDS_INSUFF_PRIV));
return EXIT_FAILURE ;
}
// Load ntdll.dll from the system directory. This is used to get the
// function addresses for the various NT Boot Entry Management API's used by
// this tool.
if(!GetSystemDirectory( dllName, MAX_PATH ))
{
ShowLastErrorEx( stderr, SLE_TYPE_ERROR | SLE_SYSTEM );
return EXIT_FAILURE ;
}
StringConcat(dllName, _T("\\ntdll.dll"), SIZE_OF_ARRAY(dllName));
hModule = LoadLibrary( dllName );
if ( hModule == NULL )
{
ShowLastErrorEx( stderr, SLE_TYPE_ERROR | SLE_SYSTEM );
return EXIT_FAILURE ;
}
// Get the system boot order list.
length = 0;
status = NtQueryBootEntryOrder( NULL, &length );
if ( status != STATUS_BUFFER_TOO_SMALL )
{
if ( status == STATUS_SUCCESS )
{
length = 0;
}
else
{
error = RtlNtStatusToDosError( status );
ShowMessage(stderr,GetResString(IDS_ERROR_QUERY_BOOTENTRY) );
return EXIT_FAILURE ;
}
}
if ( length != 0 )
{
BootEntryOrder = (PULONG)AllocateMemory( length * sizeof(ULONG) );
if(BootEntryOrder == NULL)
{
ShowLastErrorEx( stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return EXIT_FAILURE ;
}
status = NtQueryBootEntryOrder( BootEntryOrder, &length );
if ( status != STATUS_SUCCESS )
{
error = RtlNtStatusToDosError( status );
ShowMessage(stderr,GetResString(IDS_ERROR_QUERY_BOOTENTRY) );
if(BootEntryOrder)
{
FreeMemory((LPVOID *)&BootEntryOrder);
}
return EXIT_FAILURE ;
}
}
BootEntryOrderCount = length;
//Enumerate all the boot entries
status = BootCfg_EnumerateBootEntries(&ntBootEntries);
if ( status != STATUS_SUCCESS )
{
error = RtlNtStatusToDosError( status );
//free the ntBootEntries list
if(ntBootEntries)
{
FreeMemory((LPVOID *)&ntBootEntries);
}
if(BootEntryOrder)
{
FreeMemory((LPVOID *)&BootEntryOrder);
}
return EXIT_FAILURE ;
}
//Initialize the various head pointers
InitializeListHead( &BootEntries );
InitializeListHead( &ActiveUnorderedBootEntries );
InitializeListHead( &InactiveUnorderedBootEntries );
//Convert the bootentries into our know format -- MY_BOOT_ENTRIES.
if(ConvertBootEntries( ntBootEntries ) == EXIT_FAILURE)
{
if(ntBootEntries)
{
FreeMemory((LPVOID *)&ntBootEntries);
}
if(BootEntryOrder)
{
FreeMemory((LPVOID *)&BootEntryOrder);
}
return EXIT_FAILURE ;
}
//free the memory allocated for the enumeration
if(ntBootEntries)
{
FreeMemory((LPVOID *)&ntBootEntries);
}
// Build the ordered boot entry list.
myId = 1;
for ( i = 0; i < BootEntryOrderCount; i++ )
{
ULONG id = BootEntryOrder[i];
for ( listEntry = ActiveUnorderedBootEntries.Flink;
listEntry != &ActiveUnorderedBootEntries;
listEntry = listEntry->Flink )
{
bootEntry = CONTAINING_RECORD( listEntry, MY_BOOT_ENTRY, ListEntry );
if ( bootEntry->Id == id )
{
//Mark this entry as "Ordered" as the ordered id is found
bootEntry->Ordered = 1;
//Assign the internal ID
bootEntry->myId = myId++;
listEntry = listEntry->Blink;
RemoveEntryList( &bootEntry->ListEntry );
InsertTailList( &BootEntries, &bootEntry->ListEntry );
bootEntry->ListHead = &BootEntries;
}
}
for ( listEntry = InactiveUnorderedBootEntries.Flink;
listEntry != &InactiveUnorderedBootEntries;
listEntry = listEntry->Flink )
{
bootEntry = CONTAINING_RECORD( listEntry, MY_BOOT_ENTRY, ListEntry );
if ( bootEntry->Id == id )
{
//Mark this entry as ordered as the ordered id is found
bootEntry->Ordered = 1;
//Assign the internal ID
bootEntry->myId = myId++;
listEntry = listEntry->Blink;
RemoveEntryList( &bootEntry->ListEntry );
InsertTailList( &BootEntries, &bootEntry->ListEntry );
bootEntry->ListHead = &BootEntries;
}
}
}
//Now add the boot entries that are not a part of the ordered list
for (listEntry = ActiveUnorderedBootEntries.Flink;
listEntry != &ActiveUnorderedBootEntries;
listEntry = listEntry->Flink )
{
bootEntry = CONTAINING_RECORD( listEntry, MY_BOOT_ENTRY, ListEntry );
if ( bootEntry->Ordered != 1 )
{
//Assign the internal ID
bootEntry->myId = myId++;
listEntry = listEntry->Blink;
RemoveEntryList( &bootEntry->ListEntry );
InsertTailList( &BootEntries, &bootEntry->ListEntry );
bootEntry->ListHead = &BootEntries;
}
}
for (listEntry = InactiveUnorderedBootEntries.Flink;listEntry != &InactiveUnorderedBootEntries;listEntry = listEntry->Flink)
{
bootEntry = CONTAINING_RECORD( listEntry, MY_BOOT_ENTRY, ListEntry );
if ( bootEntry->Id != 1 )
{
//Assign the internal ID
bootEntry->myId = myId++;
listEntry = listEntry->Blink;
RemoveEntryList( &bootEntry->ListEntry );
InsertTailList( &BootEntries, &bootEntry->ListEntry );
bootEntry->ListHead = &BootEntries;
}
}
if(BootEntryOrder)
{
FreeMemory((LPVOID *)&BootEntryOrder);
}
return EXIT_SUCCESS ;
}
BOOL QueryBootIniSettings_IA64( DWORD argc, LPCTSTR argv[])
/*++
Name : QueryBootIniSettings_IA64
Synopsis : This routine is displays the boot entries and their settings
for an EFI based machine
Parameters : None
Return Type : VOID
Global Variables: Global Linked lists for storing the boot entries
LIST_ENTRY BootEntries;
--*/
{
// Builiding the TCMDPARSER structure
BOOL bQuery = FALSE ;
BOOL bUsage = FALSE ;
DWORD dwExitcode = 0 ;
TCMDPARSER2 cmdOptions[2];
PTCMDPARSER2 pcmdOption;
SecureZeroMemory(cmdOptions, SIZE_OF_ARRAY(cmdOptions)*sizeof(TCMDPARSER2) );
//main option
pcmdOption = &cmdOptions[0];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_QUERY;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bQuery;
//usage option
pcmdOption = &cmdOptions[1];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_USAGE;
pcmdOption->dwFlags = CP2_USAGE;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bUsage;
if ( !DoParseParam2( argc, argv, 0, SIZE_OF_ARRAY(cmdOptions ), cmdOptions, 0 ) )
{
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return (EXIT_FAILURE);
}
//check if usage is specified with more than one option
if( (TRUE == bUsage) && (argc > 3) )
{
ShowMessage(stderr,GetResString(IDS_QUERY_USAGE));
return ( EXIT_FAILURE );
}
if( bUsage )
{
displayQueryUsage();
return EXIT_SUCCESS ;
}
dwExitcode = InitializeEFI();
if(EXIT_FAILURE == dwExitcode )
{
return EXIT_FAILURE ;
}
if(DisplayBootOptions() == EXIT_FAILURE)
{
return EXIT_FAILURE;
}
DisplayBootEntry();
//Remember to free the memory for the linked lists here
Freelist();
return EXIT_SUCCESS;
}
NTSTATUS
BootCfg_EnumerateBootEntries(
PBOOT_ENTRY_LIST *ntBootEntries
)
/*++
Routine Description :
This routine enumerates the boot entries and fills the
BootEntryList
This routine will fill in the Boot entry list. The caller
of this function needs to free the memory for ntBootEntries.
Arguments
ntBootEntries : Pointer to the BOOT_ENTRY_LIST structure
Return Type : NTSTATUS
--*/
{
DWORD error;
NTSTATUS status;
ULONG length = 0;
// Query all existing boot entries.
status = NtEnumerateBootEntries( NULL, &length );
if ( status != STATUS_BUFFER_TOO_SMALL )
{
if ( status == STATUS_SUCCESS )
{
length = 0;
}
else
{
error = RtlNtStatusToDosError( status );
ShowMessage(stderr,GetResString(IDS_ERROR_ENUM_BOOTENTRY) );
}
}
if ( length != 0 )
{
*ntBootEntries = (PBOOT_ENTRY_LIST)AllocateMemory( length );
if(*ntBootEntries == NULL)
{
ShowLastErrorEx( stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return STATUS_UNSUCCESSFUL;
}
status = NtEnumerateBootEntries( *ntBootEntries, &length );
if ( status != STATUS_SUCCESS )
{
error = RtlNtStatusToDosError( status );
FreeMemory((LPVOID *)&(*ntBootEntries) );
ShowMessage(stderr,GetResString(IDS_ERROR_ENUM_BOOTENTRY) );
}
}
return status;
}
NTSTATUS
BootCfg_QueryBootOptions( IN PBOOT_OPTIONS *ppBootOptions)
/*++
Routine Description :
This routine enumerates the boot options and fills the
BOOT_OPTIONS
The caller of this function needs to free the memory for
BOOT_OPTIONS.
Arguments
ppBootOptions : Pointer to the BOOT_ENTRY_LIST structure
Return Type : NTSTATUS
--*/
{
DWORD error;
NTSTATUS status;
ULONG length = 0;
//Querying the Boot options
status = NtQueryBootOptions( NULL, &length );
if ( status == STATUS_NOT_IMPLEMENTED )
{
ShowMessage( stderr,GetResString(IDS_NO_EFINVRAM) );
return STATUS_UNSUCCESSFUL;
}
if ( status != STATUS_BUFFER_TOO_SMALL )
{
error = RtlNtStatusToDosError( status );
ShowMessage(stderr,GetResString(IDS_ERROR_QUERY_BOOTOPTIONS) );
return STATUS_UNSUCCESSFUL;
}
*ppBootOptions = (PBOOT_OPTIONS)AllocateMemory(length);
if(*ppBootOptions == NULL)
{
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return STATUS_UNSUCCESSFUL;
}
status = NtQueryBootOptions( *ppBootOptions, &length );
if ( status != STATUS_SUCCESS )
{
error = RtlNtStatusToDosError( status );
ShowMessage(stderr,GetResString(IDS_ERROR_QUERY_BOOTOPTIONS) );
FreeMemory( (LPVOID *) &*ppBootOptions );
return STATUS_UNSUCCESSFUL;
}
return status;
}
DWORD
RawStringOsOptions_IA64( IN DWORD argc,
IN LPCTSTR argv[]
)
/*++
Routine Description :
Allows the user to add the OS load options specifed
as a raw string at the cmdline to the boot
Arguments
[ in ] argc - Number of command line arguments
[ in ] argv - Array containing command line arguments
Return Type : DWORD
Returns EXIT_SUCCESS if function is successful,
returns EXIT_FAILURE otherwise.
--*/
{
BOOL bUsage = FALSE ;
BOOL bRaw = FALSE ;
DWORD dwBootID = 0;
BOOL bBootIdFound = FALSE;
DWORD dwExitCode = ERROR_SUCCESS;
PMY_BOOT_ENTRY mybootEntry;
PLIST_ENTRY listEntry;
PBOOT_ENTRY bootEntry;
STRING256 szRawString = NULL_STRING ;
BOOL bAppendFlag = FALSE ;
STRING256 szAppendString = NULL_STRING ;
PWINDOWS_OS_OPTIONS pWindowsOptions;
DWORD dwExitcode = 0 ;
TCMDPARSER2 cmdOptions[5];
PTCMDPARSER2 pcmdOption;
// Building the TCMDPARSER structure
SecureZeroMemory(cmdOptions, SIZE_OF_ARRAY(cmdOptions)*sizeof(TCMDPARSER2) );
//main option
pcmdOption = &cmdOptions[0];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_RAW;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bRaw;
//usage option
pcmdOption = &cmdOptions[1];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_USAGE;
pcmdOption->dwFlags = CP2_USAGE;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bUsage;
//id option
pcmdOption = &cmdOptions[2];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_ID;
pcmdOption->dwFlags = CP_VALUE_MANDATORY | CP2_MANDATORY;
pcmdOption->dwType = CP_TYPE_UNUMERIC;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &dwBootID;
//default option
pcmdOption = &cmdOptions[3];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_DEFAULT;
pcmdOption->dwFlags = CP2_DEFAULT | CP2_MANDATORY | CP2_VALUE_TRIMINPUT;
pcmdOption->dwType = CP_TYPE_TEXT;
pcmdOption->dwCount = 1;
pcmdOption->pValue = szRawString;
pcmdOption->dwLength= MAX_STRING_LENGTH;
//append option
pcmdOption = &cmdOptions[4];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_APPEND;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bAppendFlag;
// Parsing the copy option switches
if ( !DoParseParam2( argc, argv, 0, SIZE_OF_ARRAY(cmdOptions ), cmdOptions, 0 ) )
{
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return (EXIT_FAILURE);
}
//check if usage is specified with more than one option
if( (TRUE == bUsage) && (argc > 3) )
{
ShowMessage(stderr,GetResString(IDS_RAW_USAGE));
return ( EXIT_FAILURE );
}
// Displaying query usage if user specified -? with -query option
if( bUsage )
{
displayRawUsage_IA64();
return (EXIT_SUCCESS);
}
dwExitcode = InitializeEFI();
if(EXIT_FAILURE == dwExitcode )
{
return EXIT_FAILURE ;
}
//Trim any leading or trailing spaces
if(StringLengthW(szRawString,0) != 0)
{
TrimString(szRawString, TRIM_ALL);
}
//Query the boot entries till u get the BootID specified by the user
for (listEntry = BootEntries.Flink;listEntry != &BootEntries;listEntry = listEntry->Flink)
{
//Get the boot entry
mybootEntry = CONTAINING_RECORD( listEntry, MY_BOOT_ENTRY, ListEntry );
if(mybootEntry->myId == dwBootID)
{
bBootIdFound = TRUE;
bootEntry = &mybootEntry->NtBootEntry;
//Check whether the bootEntry is a Windows one or not.
//The OS load options can be added only to a Windows boot entry.
if(!IsBootEntryWindows(bootEntry))
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_OSOPTIONS),dwBootID);
dwExitCode = EXIT_FAILURE;
break;
}
pWindowsOptions = (PWINDOWS_OS_OPTIONS)bootEntry->OsOptions;
if(bAppendFlag == TRUE )
{
StringCopy(szAppendString,pWindowsOptions->OsLoadOptions, SIZE_OF_ARRAY(szAppendString));
StringConcat(szAppendString,TOKEN_EMPTYSPACE, SIZE_OF_ARRAY(szAppendString) );
StringConcat(szAppendString,szRawString, SIZE_OF_ARRAY(szAppendString));
}
else
{
StringCopy(szAppendString,szRawString, SIZE_OF_ARRAY(szAppendString));
}
//display error message if Os Load options is more than 254
// characters.
if(StringLengthW(szAppendString, 0) > MAX_RES_STRING)
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_STRING_LENGTH1),MAX_RES_STRING);
return EXIT_FAILURE ;
}
//
//Change the OS load options.
//Pass NULL to friendly name as we are not changing the same
//szAppendString is the Os load options specified by the user
//to be appended or to be overwritten over the existing options
//
dwExitCode = ChangeBootEntry(bootEntry, NULL, szAppendString);
if(dwExitCode == ERROR_SUCCESS)
{
ShowMessageEx(stdout, 1, TRUE, GetResString(IDS_SUCCESS_OSOPTIONS),dwBootID);
}
else
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_OSOPTIONS),dwBootID);
}
break;
}
}
if(bBootIdFound == FALSE)
{
//Could not find the BootID specified by the user so output the message and return failure
ShowMessage(stderr,GetResString(IDS_INVALID_BOOTID));
dwExitCode = EXIT_FAILURE;
}
//Remember to free memory allocated for the linked lists
Freelist();
return (dwExitCode);
}
DWORD
ChangeBootEntry( IN PBOOT_ENTRY bootEntry,
IN LPTSTR lpNewFriendlyName,
IN LPTSTR lpOSLoadOptions)
/*++
Routine Description:
This routine is used to change the FriendlyName and the
OS Options for a boot entry.
Arguments
[ in ] bootEntry -Pointer to a BootEntry structure
for which the changes needs to be made
[ in ] lpNewFriendlyName -String specifying the new friendly name.
[ in ] lpOSLoadOptions - String specifying the OS load options.
Return Type : DWORD -- ERROR_SUCCESS on success
-- ERROR_FAILURE on failure
--*/
{
PBOOT_ENTRY bootEntryCopy;
PMY_BOOT_ENTRY myBootEntry;
PWINDOWS_OS_OPTIONS osOptions;
ULONG length;
PMY_BOOT_ENTRY myChBootEntry;
NTSTATUS status;
DWORD error, dwErrorCode = ERROR_SUCCESS;
// Calculate the length of our internal structure. This includes
// the base part of MY_BOOT_ENTRY plus the NT BOOT_ENTRY.
//
length = FIELD_OFFSET(MY_BOOT_ENTRY, NtBootEntry) + bootEntry->Length;
myBootEntry = (PMY_BOOT_ENTRY)AllocateMemory(length);
if( NULL == myBootEntry )
{
ShowLastErrorEx( stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return EXIT_FAILURE;
}
RtlZeroMemory(myBootEntry, length);
//
// Copy the NT BOOT_ENTRY into the allocated buffer.
//
bootEntryCopy = &myBootEntry->NtBootEntry;
memcpy(bootEntryCopy, bootEntry, bootEntry->Length);
myBootEntry->Id = bootEntry->Id;
myBootEntry->Attributes = bootEntry->Attributes;
//Change the friendly name if lpNewFriendlyName is not NULL
if(lpNewFriendlyName)
{
myBootEntry->FriendlyName = lpNewFriendlyName;
myBootEntry->FriendlyNameLength = ((ULONG)StringLengthW(lpNewFriendlyName, 0) + 1) * sizeof(WCHAR);
}
else
{
myBootEntry->FriendlyName = (PWSTR)ADD_OFFSET(bootEntryCopy, FriendlyNameOffset);
myBootEntry->FriendlyNameLength = ((ULONG)StringLengthW(myBootEntry->FriendlyName, 0) + 1) * sizeof(WCHAR);
}
myBootEntry->BootFilePath = (PFILE_PATH)ADD_OFFSET(bootEntryCopy, BootFilePathOffset);
// If this is an NT boot entry, capture the NT-specific information in
// the OsOptions.
osOptions = (PWINDOWS_OS_OPTIONS)bootEntryCopy->OsOptions;
if ((bootEntryCopy->OsOptionsLength >= FIELD_OFFSET(WINDOWS_OS_OPTIONS, OsLoadOptions)) &&
(strcmp((char *)osOptions->Signature, WINDOWS_OS_OPTIONS_SIGNATURE) == 0))
{
MBE_SET_IS_NT( myBootEntry );
//To change the OS Load options
if(lpOSLoadOptions)
{
myBootEntry->OsLoadOptions = lpOSLoadOptions;
myBootEntry->OsLoadOptionsLength = ((ULONG)StringLengthW(lpOSLoadOptions, 0) + 1) * sizeof(WCHAR);
}
else
{
myBootEntry->OsLoadOptions = osOptions->OsLoadOptions;
myBootEntry->OsLoadOptionsLength = ((ULONG)StringLengthW(myBootEntry->OsLoadOptions, 0) + 1) * sizeof(WCHAR);
}
myBootEntry->OsFilePath = (PFILE_PATH)ADD_OFFSET(osOptions, OsLoadPathOffset);
}
else
{
// Foreign boot entry. Just capture whatever OS options exist.
//
myBootEntry->ForeignOsOptions = bootEntryCopy->OsOptions;
myBootEntry->ForeignOsOptionsLength = bootEntryCopy->OsOptionsLength;
}
myChBootEntry = CreateBootEntryFromBootEntry(myBootEntry);
if(myChBootEntry == NULL)
{
ShowLastErrorEx( stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
if(myBootEntry)
{
FreeMemory((LPVOID *)&myBootEntry);
}
return EXIT_FAILURE;
}
//Call the modify API
status = NtModifyBootEntry(&myChBootEntry->NtBootEntry);
if ( status != STATUS_SUCCESS )
{
error = RtlNtStatusToDosError( status );
dwErrorCode = error;
ShowMessage(stderr,GetResString(IDS_ERROR_MODIFY_BOOTENTRY) );
}
//free the memory
if(myChBootEntry)
{
FreeMemory((LPVOID *)&myChBootEntry);
}
if(myBootEntry)
{
FreeMemory((LPVOID *)&myBootEntry);
}
return dwErrorCode;
}
PMY_BOOT_ENTRY
CreateBootEntryFromBootEntry (IN PMY_BOOT_ENTRY OldBootEntry)
/*++
Routine Description :
This routine is used to create a new MY_BOOT_ENTRY struct.
The caller of this function needs to free the memory allocated
for the MY_BOOT_ENTRY struct.
Arguments :
[ in ] bootEntry - Pointer to a BootEntry structure
for which the changes needs to be made
[ in ] lpNewFriendlyName - String specifying the new friendly name.
[ in ] lpOSLoadOptions - String specifying the OS load options.
Return Type : PMY_BOOT_ENTRY - Pointer to the new MY_BOOT_ENTRY strucure.
NULL on failure
--*/
{
ULONG requiredLength;
ULONG osOptionsOffset;
ULONG osLoadOptionsLength;
ULONG osLoadPathOffset;
ULONG osLoadPathLength;
ULONG osOptionsLength;
ULONG friendlyNameOffset;
ULONG friendlyNameLength;
ULONG bootPathOffset;
ULONG bootPathLength;
PMY_BOOT_ENTRY newBootEntry;
PBOOT_ENTRY ntBootEntry;
PWINDOWS_OS_OPTIONS osOptions;
PFILE_PATH osLoadPath;
PWSTR friendlyName;
PFILE_PATH bootPath;
// Calculate how long the internal boot entry needs to be. This includes
// our internal structure, plus the BOOT_ENTRY structure that the NT APIs
// use.
//
// Our structure:
//
requiredLength = FIELD_OFFSET(MY_BOOT_ENTRY, NtBootEntry);
// Base part of NT structure:
//
requiredLength += FIELD_OFFSET(BOOT_ENTRY, OsOptions);
// Save offset to BOOT_ENTRY.OsOptions. Add in base part of
// WINDOWS_OS_OPTIONS. Calculate length in bytes of OsLoadOptions
// and add that in.
//
osOptionsOffset = requiredLength;
if ( MBE_IS_NT( OldBootEntry ) )
{
// Add in base part of WINDOWS_OS_OPTIONS. Calculate length in
// bytes of OsLoadOptions and add that in.
//
requiredLength += FIELD_OFFSET(WINDOWS_OS_OPTIONS, OsLoadOptions);
osLoadOptionsLength = OldBootEntry->OsLoadOptionsLength;
requiredLength += osLoadOptionsLength;
// Round up to a ULONG boundary for the OS FILE_PATH in the
// WINDOWS_OS_OPTIONS. Save offset to OS FILE_PATH. Calculate length
// in bytes of FILE_PATH and add that in. Calculate total length of
// WINDOWS_OS_OPTIONS.
//
requiredLength = ALIGN_UP(requiredLength, ULONG);
osLoadPathOffset = requiredLength;
requiredLength += OldBootEntry->OsFilePath->Length;
osLoadPathLength = requiredLength - osLoadPathOffset;
}
else
{
// Add in length of foreign OS options.
//
requiredLength += OldBootEntry->ForeignOsOptionsLength;
osLoadOptionsLength = 0;
osLoadPathOffset = 0;
osLoadPathLength = 0;
}
osOptionsLength = requiredLength - osOptionsOffset;
// Round up to a ULONG boundary for the friendly name in the BOOT_ENTRY.
// Save offset to friendly name. Calculate length in bytes of friendly name
// and add that in.
//
requiredLength = ALIGN_UP(requiredLength, ULONG);
friendlyNameOffset = requiredLength;
friendlyNameLength = OldBootEntry->FriendlyNameLength;
requiredLength += friendlyNameLength;
// Round up to a ULONG boundary for the boot FILE_PATH in the BOOT_ENTRY.
// Save offset to boot FILE_PATH. Calculate length in bytes of FILE_PATH
// and add that in.
//
requiredLength = ALIGN_UP(requiredLength, ULONG);
bootPathOffset = requiredLength;
requiredLength += OldBootEntry->BootFilePath->Length;
bootPathLength = requiredLength - bootPathOffset;
// Allocate memory for the boot entry.
//
newBootEntry = (PMY_BOOT_ENTRY)AllocateMemory(requiredLength);
if(newBootEntry == NULL)
{
return NULL;
}
RtlZeroMemory(newBootEntry, requiredLength);
// Calculate addresses of various substructures using the saved offsets.
//
ntBootEntry = &newBootEntry->NtBootEntry;
osOptions = (PWINDOWS_OS_OPTIONS)ntBootEntry->OsOptions;
osLoadPath = (PFILE_PATH)((PUCHAR)newBootEntry + osLoadPathOffset);
friendlyName = (PWSTR)((PUCHAR)newBootEntry + friendlyNameOffset);
bootPath = (PFILE_PATH)((PUCHAR)newBootEntry + bootPathOffset);
// Fill in the internal-format structure.
//
// newBootEntry->AllocationEnd = (PUCHAR)newBootEntry + requiredLength;
newBootEntry->Status = OldBootEntry->Status & MBE_STATUS_IS_NT;
newBootEntry->Attributes = OldBootEntry->Attributes;
newBootEntry->Id = OldBootEntry->Id;
newBootEntry->FriendlyName = friendlyName;
newBootEntry->FriendlyNameLength = friendlyNameLength;
newBootEntry->BootFilePath = bootPath;
if ( MBE_IS_NT( OldBootEntry ) )
{
newBootEntry->OsLoadOptions = osOptions->OsLoadOptions;
newBootEntry->OsLoadOptionsLength = osLoadOptionsLength;
newBootEntry->OsFilePath = osLoadPath;
}
// Fill in the base part of the NT boot entry.
//
ntBootEntry->Version = BOOT_ENTRY_VERSION;
ntBootEntry->Length = requiredLength - FIELD_OFFSET(MY_BOOT_ENTRY, NtBootEntry);
ntBootEntry->Attributes = OldBootEntry->Attributes;
ntBootEntry->Id = OldBootEntry->Id;
ntBootEntry->FriendlyNameOffset = (ULONG)((PUCHAR)friendlyName - (PUCHAR)ntBootEntry);
ntBootEntry->BootFilePathOffset = (ULONG)((PUCHAR)bootPath - (PUCHAR)ntBootEntry);
ntBootEntry->OsOptionsLength = osOptionsLength;
if ( MBE_IS_NT( OldBootEntry ) )
{
// Fill in the base part of the WINDOWS_OS_OPTIONS, including the
// OsLoadOptions.
//
StringCopyA((char *)osOptions->Signature, WINDOWS_OS_OPTIONS_SIGNATURE, sizeof(WINDOWS_OS_OPTIONS_SIGNATURE));
osOptions->Version = WINDOWS_OS_OPTIONS_VERSION;
osOptions->Length = osOptionsLength;
osOptions->OsLoadPathOffset = (ULONG)((PUCHAR)osLoadPath - (PUCHAR)osOptions);
StringCopy(osOptions->OsLoadOptions, OldBootEntry->OsLoadOptions, osOptions->Length );
// Copy the OS FILE_PATH.
//
memcpy( osLoadPath, OldBootEntry->OsFilePath, osLoadPathLength );
}
else
{
// Copy the foreign OS options.
memcpy( osOptions, OldBootEntry->ForeignOsOptions, osOptionsLength );
}
// Copy the friendly name.
StringCopy(friendlyName, OldBootEntry->FriendlyName, friendlyNameLength );
// Copy the boot FILE_PATH.
memcpy( bootPath, OldBootEntry->BootFilePath, bootPathLength );
return newBootEntry;
} // CreateBootEntryFromBootEntry
DWORD
DeleteBootIniSettings_IA64( IN DWORD argc,
IN LPCTSTR argv[]
)
/*++
//
// Routine Description : This routine deletes an existing boot entry from an EFI
// based machine
//
// Arguments :
// [ in ] argc - Number of command line arguments
// [ in ] argv - Array containing command line arguments
//
// Return Type : DWORD
// Returns EXIT_SUCCESS if successful,
// returns EXIT_FAILURE otherwise.
//
//
--*/
{
BOOL bDelete = FALSE ;
BOOL bUsage = FALSE;
DWORD dwBootID = 0;
BOOL bBootIdFound = FALSE;
DWORD dwExitCode = ERROR_SUCCESS;
NTSTATUS status;
PMY_BOOT_ENTRY mybootEntry;
PLIST_ENTRY listEntry;
DWORD dwExitcode = 0 ;
TCMDPARSER2 cmdOptions[3];
PTCMDPARSER2 pcmdOption;
SecureZeroMemory(cmdOptions, SIZE_OF_ARRAY(cmdOptions)*sizeof(TCMDPARSER2) );
// Building the TCMDPARSER structure
//main option
pcmdOption = &cmdOptions[0];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_DELETE;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bDelete;
pcmdOption = &cmdOptions[1];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_USAGE;
pcmdOption->dwFlags = CP2_USAGE;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bUsage;
pcmdOption = &cmdOptions[2];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_ID;
pcmdOption->dwFlags = CP_VALUE_MANDATORY | CP2_MANDATORY;
pcmdOption->dwType = CP_TYPE_UNUMERIC;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &dwBootID;
// Parsing the delete option switches
if ( !DoParseParam2( argc, argv, 0, SIZE_OF_ARRAY(cmdOptions ), cmdOptions, 0 ) )
{
ShowLastErrorEx( stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return (EXIT_FAILURE);
}
//check if usage is specified with more than one option
if( (TRUE == bUsage) && (argc > 3) )
{
ShowMessage(stderr,GetResString(IDS_DELETE_USAGE));
return ( EXIT_FAILURE );
}
// Displaying delete usage if user specified -? with -delete option
if( bUsage )
{
displayDeleteUsage_IA64();
return EXIT_SUCCESS;
}
dwExitcode = InitializeEFI();
if(EXIT_FAILURE == dwExitcode )
{
return EXIT_FAILURE ;
}
//Query the boot entries till u get the BootID specified by the user
for (listEntry = BootEntries.Flink;listEntry != &BootEntries;listEntry = listEntry->Flink)
{
//
//display an error message if there is only 1 boot entry saying
//that it cannot be deleted.
//
if (listEntry->Flink == NULL)
{
ShowMessage(stderr,GetResString(IDS_ONLY_ONE_OS));
dwExitCode = EXIT_FAILURE;
break ;
}
//Get the boot entry
mybootEntry = CONTAINING_RECORD( listEntry, MY_BOOT_ENTRY, ListEntry );
if(mybootEntry->myId == dwBootID)
{
bBootIdFound = TRUE;
//Delete the boot entry specified by the user.
status = NtDeleteBootEntry(mybootEntry->Id);
if(status == STATUS_SUCCESS)
{
ShowMessageEx(stdout, 1, TRUE, GetResString(IDS_DELETE_SUCCESS),dwBootID);
}
else
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_DELETE_FAILURE),dwBootID);
}
break;
}
}
if(bBootIdFound == FALSE)
{
//Could not find the BootID specified by the user so output the message and return failure
ShowMessage(stderr,GetResString(IDS_INVALID_BOOTID));
dwExitCode = EXIT_FAILURE;
}
//Remember to free the memory allocated to the linked lists
Freelist();
return (dwExitCode);
}
BOOL
IsBootEntryWindows(PBOOT_ENTRY bootEntry)
/*++
//
//
// Routine Description :
// Checks whether the boot entry is a Windows or a foreign one
//
// Arguments :
// [ in ] bootEntry - Boot entry structure describing the
// boot entry.
//
// Return Type : BOOL
// TRUE if bootEntry is an windows entry,
// FALSE otherwise
//
--*/
{
PWINDOWS_OS_OPTIONS osOptions;
osOptions = (PWINDOWS_OS_OPTIONS)bootEntry->OsOptions;
if ((bootEntry->OsOptionsLength >= FIELD_OFFSET(WINDOWS_OS_OPTIONS, OsLoadOptions)) &&
(strcmp((char *)osOptions->Signature, WINDOWS_OS_OPTIONS_SIGNATURE) == 0))
{
return TRUE;
}
return FALSE;
}
PWSTR GetNtNameForFilePath(IN PFILE_PATH FilePath)
/*++
Routine Description :
Converts the FilePath into a NT file path.
Arguments :
[ in ] FilePath - The File path.
Return Type : PWSTR
The NT file path.
--*/
{
NTSTATUS status;
ULONG length;
PFILE_PATH ntPath;
PWSTR osDeviceNtName;
PWSTR osDirectoryNtName;
PWSTR fullNtName;
length = 0;
status = NtTranslateFilePath(
FilePath,
FILE_PATH_TYPE_NT,
NULL,
&length
);
if ( status != STATUS_BUFFER_TOO_SMALL )
{
return NULL;
}
ntPath = (PFILE_PATH)AllocateMemory( length );
if(ntPath == NULL)
{
return NULL;
}
status = NtTranslateFilePath(
FilePath,
FILE_PATH_TYPE_NT,
ntPath,
&length
);
if ( !NT_SUCCESS(status) )
{
if(ntPath)
{
FreeMemory((LPVOID *)&ntPath);
}
return NULL;
}
osDeviceNtName = (PWSTR)ntPath->FilePath;
osDirectoryNtName = osDeviceNtName + StringLengthW(osDeviceNtName,0) + 1;
length = (ULONG)(StringLengthW(osDeviceNtName,0) + StringLengthW(osDirectoryNtName, 0) + 1) * sizeof(WCHAR);
fullNtName = (PWSTR)AllocateMemory( length );
if(fullNtName == NULL)
{
if(ntPath)
{
FreeMemory((LPVOID *)&ntPath);
}
return NULL;
}
StringCopy( fullNtName, osDeviceNtName, GetBufferSize(fullNtName)/sizeof(WCHAR) );
StringConcat( fullNtName, osDirectoryNtName, GetBufferSize(fullNtName)/sizeof(WCHAR) );
if(ntPath)
{
FreeMemory((LPVOID *) &ntPath );
}
return fullNtName;
} // GetNtNameForFilePath
DWORD
CopyBootIniSettings_IA64( IN DWORD argc,
IN LPCTSTR argv[]
)
/*++
Routine Description :
This routine copies and existing boot entry for an EFI
based machine. The user can then add the various OS load
options.
Arguments :
[ in ] argc - Number of command line arguments
[ in ] argv - Array containing command line arguments
Return Type : DWORD
Returns EXIT_SUCCESS if successful,
returns EXIT_FAILURE otherwise
--*/
{
BOOL bCopy = FALSE ;
BOOL bUsage = FALSE;
DWORD dwExitCode = EXIT_SUCCESS;
DWORD dwBootID = 0;
BOOL bBootIdFound = FALSE;
PMY_BOOT_ENTRY mybootEntry;
PLIST_ENTRY listEntry;
PBOOT_ENTRY bootEntry;
DWORD dwExitcode = 0 ;
STRING256 szDescription = NULL_STRING;
BOOL bFlag = FALSE ;
TCMDPARSER2 cmdOptions[4];
PTCMDPARSER2 pcmdOption;
SecureZeroMemory(cmdOptions, SIZE_OF_ARRAY(cmdOptions)*sizeof(TCMDPARSER2) );
//main option
pcmdOption = &cmdOptions[0];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_COPY;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bCopy;
//description option
pcmdOption = &cmdOptions[1];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_DESCRIPTION;
pcmdOption->dwFlags = CP_VALUE_MANDATORY | CP2_VALUE_TRIMINPUT;
pcmdOption->dwType = CP_TYPE_TEXT;
pcmdOption->dwCount = 1;
pcmdOption->pValue = szDescription;
pcmdOption->dwLength= FRIENDLY_NAME_LENGTH;
// usage
pcmdOption = &cmdOptions[2];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_USAGE;
pcmdOption->dwFlags = CP2_USAGE;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bUsage;
//id option
pcmdOption = &cmdOptions[3];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_ID;
pcmdOption->dwFlags = CP_VALUE_MANDATORY | CP2_MANDATORY;
pcmdOption->dwType = CP_TYPE_UNUMERIC;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &dwBootID;
// Parsing the copy option switches
if ( !DoParseParam2( argc, argv, 0, SIZE_OF_ARRAY(cmdOptions ), cmdOptions, 0 ) )
{
ShowLastErrorEx( stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return (EXIT_FAILURE);
}
//check if usage is specified with more than one option
if( (TRUE == bUsage) && (argc > 3) )
{
ShowMessage(stderr,GetResString(IDS_COPY_USAGE));
return ( EXIT_FAILURE );
}
// Displaying copy usage if user specified -? with -copy option
if( bUsage )
{
displayCopyUsage_IA64();
dwExitCode = EXIT_SUCCESS;
return dwExitCode;
}
dwExitcode = InitializeEFI();
if(EXIT_FAILURE == dwExitcode )
{
return EXIT_FAILURE ;
}
if(cmdOptions[1].dwActuals != 0)
{
bFlag = TRUE ;
}
//Query the boot entries till u get the BootID specified by the user
for (listEntry = BootEntries.Flink;listEntry != &BootEntries; listEntry = listEntry->Flink)
{
//Get the boot entry
mybootEntry = CONTAINING_RECORD( listEntry, MY_BOOT_ENTRY, ListEntry );
if(mybootEntry->myId == dwBootID)
{
bBootIdFound = TRUE;
bootEntry = &mybootEntry->NtBootEntry;
//Copy the boot entry specified by the user.
dwExitCode = CopyBootEntry(bootEntry, szDescription,bFlag);
if(dwExitCode == EXIT_SUCCESS)
{
ShowMessageEx(stdout, 1, TRUE, GetResString(IDS_COPY_SUCCESS),dwBootID);
}
else
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_COPY_ERROR),dwBootID);
return EXIT_FAILURE ;
}
break;
}
}
if(bBootIdFound == FALSE)
{
//Could not find the BootID specified by the user so output the message and return failure
ShowMessage(stderr,GetResString(IDS_INVALID_BOOTID));
return (EXIT_FAILURE) ;
}
//Remember to free the memory allocated for the linked lists
Freelist();
return EXIT_SUCCESS;
}
DWORD
CopyBootEntry( IN PBOOT_ENTRY bootEntry,
IN LPTSTR lpNewFriendlyName,
IN BOOL bFlag)
/*++
Routine Description :
This routine is used to add / copy a boot entry.
Arguments :
[ in ] bootEntry - Pointer to a BootEntry structure for which the changes needs to be made
[ in ] lpNewFriendlyName - String specifying the new friendly name.
Return Type : DWORD -- ERROR_SUCCESS on success
-- EXIT_FAILURE on failure
--*/
{
PBOOT_ENTRY bootEntryCopy;
PMY_BOOT_ENTRY myBootEntry;
PWINDOWS_OS_OPTIONS osOptions;
ULONG length, Id;
PMY_BOOT_ENTRY myChBootEntry;
NTSTATUS status;
DWORD error, dwErrorCode = ERROR_SUCCESS;
WCHAR szString[500] ;
PULONG BootEntryOrder, NewBootEntryOrder, NewTempBootEntryOrder;
// Calculate the length of our internal structure. This includes
// the base part of MY_BOOT_ENTRY plus the NT BOOT_ENTRY.
//
length = FIELD_OFFSET(MY_BOOT_ENTRY, NtBootEntry) + bootEntry->Length;
myBootEntry = (PMY_BOOT_ENTRY)AllocateMemory(length);
if(myBootEntry == NULL)
{
ShowLastErrorEx( stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return (EXIT_FAILURE);
}
RtlZeroMemory(myBootEntry, length);
//
// Copy the NT BOOT_ENTRY into the allocated buffer.
//
bootEntryCopy = &myBootEntry->NtBootEntry;
memcpy(bootEntryCopy, bootEntry, bootEntry->Length);
myBootEntry->Id = bootEntry->Id;
myBootEntry->Attributes = bootEntry->Attributes;
//Change the friendly name if lpNewFriendlyName is not NULL
//if(lpNewFriendlyName && (lstrlen(lpNewFriendlyName) != 0))
//if(( cmdOptions[4].dwActuals == 0) )
if(TRUE == bFlag)
//if(lstrlen(lpNewFriendlyName) != 0)
{
myBootEntry->FriendlyName = lpNewFriendlyName;
myBootEntry->FriendlyNameLength = ((ULONG)StringLengthW(lpNewFriendlyName,0) + 1) * sizeof(WCHAR);
}
else
{
StringCopy(szString,GetResString(IDS_COPY_OF), SIZE_OF_ARRAY(szString));
StringConcat(szString,(PWSTR)ADD_OFFSET(bootEntryCopy, FriendlyNameOffset), SIZE_OF_ARRAY(szString));
if(StringLengthW(szString, 0) >= 67)
{
StringCopy(szString,szString,67);
}
myBootEntry->FriendlyName = szString ;
myBootEntry->FriendlyNameLength = ((ULONG)StringLengthW(szString,0) + 1) * sizeof(WCHAR);
}
myBootEntry->BootFilePath = (PFILE_PATH)ADD_OFFSET(bootEntryCopy, BootFilePathOffset);
// If this is an NT boot entry, capture the NT-specific information in
// the OsOptions.
osOptions = (PWINDOWS_OS_OPTIONS)bootEntryCopy->OsOptions;
if ((bootEntryCopy->OsOptionsLength >= FIELD_OFFSET(WINDOWS_OS_OPTIONS, OsLoadOptions)) &&
(strcmp((char *)osOptions->Signature, WINDOWS_OS_OPTIONS_SIGNATURE) == 0))
{
MBE_SET_IS_NT( myBootEntry );
//To change the OS Load options
myBootEntry->OsLoadOptions = osOptions->OsLoadOptions;
myBootEntry->OsLoadOptionsLength = ((ULONG)StringLengthW(myBootEntry->OsLoadOptions, 0) + 1) * sizeof(WCHAR);
myBootEntry->OsFilePath = (PFILE_PATH)ADD_OFFSET(osOptions, OsLoadPathOffset);
}
else
{
// Foreign boot entry. Just capture whatever OS options exist.
//
myBootEntry->ForeignOsOptions = bootEntryCopy->OsOptions;
myBootEntry->ForeignOsOptionsLength = bootEntryCopy->OsOptionsLength;
}
myChBootEntry = CreateBootEntryFromBootEntry(myBootEntry);
if(myChBootEntry == NULL)
{
ShowLastErrorEx( stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
if(myBootEntry)
{
FreeMemory((LPVOID *)&myBootEntry);
}
ShowLastError(stderr);
return (EXIT_FAILURE);
}
//Call the NtAddBootEntry API
status = NtAddBootEntry(&myChBootEntry->NtBootEntry, &Id);
if ( status != STATUS_SUCCESS )
{
error = RtlNtStatusToDosError( status );
dwErrorCode = error;
ShowMessage(stderr,GetResString(IDS_ERROR_UNEXPECTED) );
}
// Get the system boot order list.
length = 0;
status = NtQueryBootEntryOrder( NULL, &length );
if ( status != STATUS_BUFFER_TOO_SMALL )
{
if ( status == STATUS_SUCCESS )
{
length = 0;
}
else
{
error = RtlNtStatusToDosError( status );
ShowMessage(stderr,GetResString(IDS_ERROR_QUERY_BOOTENTRY) );
if(myBootEntry)
{
FreeMemory((LPVOID *)&myBootEntry);
}
if(myChBootEntry)
{
FreeMemory((LPVOID *)&myChBootEntry);
}
return FALSE;
}
}
if ( length != 0 )
{
BootEntryOrder = (PULONG)AllocateMemory( length * sizeof(ULONG) );
if(BootEntryOrder == NULL)
{
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
if(myBootEntry)
{
FreeMemory((LPVOID *)&myBootEntry);
}
if(myChBootEntry)
{
FreeMemory((LPVOID *)&myChBootEntry);
}
return (EXIT_FAILURE);
}
status = NtQueryBootEntryOrder( BootEntryOrder, &length );
if ( status != STATUS_SUCCESS )
{
error = RtlNtStatusToDosError( status );
ShowMessage(stderr,GetResString(IDS_ERROR_QUERY_BOOTENTRY));
dwErrorCode = error;
FreeMemory((LPVOID *)&myBootEntry);
FreeMemory((LPVOID *)&BootEntryOrder);
FreeMemory((LPVOID *)&myChBootEntry);
return dwErrorCode;
}
}
//Allocate memory for the new boot entry order.
NewBootEntryOrder = (PULONG)AllocateMemory((length+1) * sizeof(ULONG));
if(NULL == NewBootEntryOrder )
{
ShowLastErrorEx( stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
FreeMemory((LPVOID *)&myBootEntry);
FreeMemory((LPVOID *)&BootEntryOrder);
FreeMemory((LPVOID *)&myChBootEntry);
return (EXIT_FAILURE);
}
NewTempBootEntryOrder = NewBootEntryOrder;
memcpy(NewTempBootEntryOrder,BootEntryOrder,length*sizeof(ULONG));
NewTempBootEntryOrder = NewTempBootEntryOrder + length;
*NewTempBootEntryOrder = Id;
status = NtSetBootEntryOrder(NewBootEntryOrder, length+1);
if ( status != STATUS_SUCCESS )
{
error = RtlNtStatusToDosError( status );
dwErrorCode = error;
ShowMessage(stderr,GetResString(IDS_ERROR_SET_BOOTENTRY));
}
//free the memory
FreeMemory((LPVOID *)&NewBootEntryOrder);
FreeMemory((LPVOID *)&myBootEntry);
FreeMemory((LPVOID *)&BootEntryOrder);
FreeMemory((LPVOID *)&myChBootEntry);
return dwErrorCode;
}
DWORD
ChangeTimeOut_IA64( IN DWORD argc,
IN LPCTSTR argv[]
)
/*++
Routine Description :
This routine chnages the Timeout value in the system
global boot options.
Arguments :
[ in ] argc - Number of command line arguments
[ in ] argv - Array containing command line arguments
Return Type : DOWRD
Returns EXIT_SUCCESS if it successfull,
returns EXIT_FAILURE otherwise.
--*/
{
DWORD dwTimeOut = 0;
DWORD dwExitCode = EXIT_SUCCESS;
ULONG Flag = 0;
DWORD dwExitcode = 0;
BOOL bTimeout = FALSE;
BOOL bUsage = FALSE;
TCMDPARSER2 cmdOptions[3];
PTCMDPARSER2 pcmdOption;
SecureZeroMemory(cmdOptions, SIZE_OF_ARRAY(cmdOptions)*sizeof(TCMDPARSER2) );
//main option
pcmdOption = &cmdOptions[0];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_TIMEOUT;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bTimeout;
//default timeout value
pcmdOption = &cmdOptions[1];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_DEFAULT;
pcmdOption->dwFlags = CP2_DEFAULT | CP2_MANDATORY;
pcmdOption->dwType = CP_TYPE_UNUMERIC;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &dwTimeOut;
//usage option
pcmdOption = &cmdOptions[2];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_USAGE;
pcmdOption->dwFlags = CP2_USAGE;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bUsage;
if( !DoParseParam2( argc, argv, 0, SIZE_OF_ARRAY(cmdOptions ), cmdOptions, 0 ) )
{
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return (EXIT_FAILURE);
}
//check if usage is specified with more than one option
if( (TRUE == bUsage) && (argc > 3) )
{
ShowMessage(stderr,GetResString(IDS_TIMEOUT_USAGE));
return ( EXIT_FAILURE );
}
if(bUsage)
{
displayTimeOutUsage_IA64();
return (EXIT_SUCCESS);
}
dwExitcode = InitializeEFI();
if(EXIT_FAILURE == dwExitcode )
{
return EXIT_FAILURE ;
}
//Check for the limit of Timeout value entered by the user.
if(dwTimeOut > TIMEOUT_MAX )
{
ShowMessage(stderr,GetResString(IDS_TIMEOUT_RANGE));
return (EXIT_FAILURE );
}
//Call the ModifyBootOptions function with the BOOT_OPTIONS_FIELD_COUNTDOWN
Flag |= BOOT_OPTIONS_FIELD_COUNTDOWN;
dwExitCode = ModifyBootOptions(dwTimeOut, NULL, 0, Flag);
return dwExitCode;
}
DWORD
ModifyBootOptions( IN ULONG Timeout,
IN LPTSTR pHeadlessRedirection,
IN ULONG NextBootEntryID,
IN ULONG Flag
)
/*++
Routine Description :
This routine Modifies the Boot options
-Timeout
-NextBootEntryID
-HeadlessRedirection
Arguments :
[ in ] Timeout - The new Timeout value
[ in ] pHeadlessRedirection - The Headless redirection string
[ in ] NextBootEntryID - The NextBootEntryID
[ in ] Flag - The Flags indicating what fields that needs to be changed
BOOT_OPTIONS_FIELD_COUNTDOWN
BOOT_OPTIONS_FIELD_NEXT_BOOT_ENTRY_ID
BOOT_OPTIONS_FIELD_HEADLESS_REDIRECTION
Return Type : DOWRD
Returns EXIT_SUCCESS if successful,
returns EXIT_FAILURE otherwise.
--*/
{
PBOOT_OPTIONS pBootOptions;
PBOOT_OPTIONS pModifiedBootOptions;
DWORD error;
NTSTATUS status;
ULONG newlength=0;
DWORD dwExitCode = EXIT_SUCCESS;
NextBootEntryID = 0;
//Query the existing Boot options and modify based on the Flag value
status = BootCfg_QueryBootOptions(&pBootOptions);
if(status != STATUS_SUCCESS)
{
error = RtlNtStatusToDosError( status );
FreeMemory((LPVOID *)&pBootOptions);
return (error);
}
//Calculate the new length of the BOOT_OPTIONS struct based on the fields that needs to be changed.
newlength = FIELD_OFFSET(BOOT_OPTIONS, HeadlessRedirection);
if((Flag & BOOT_OPTIONS_FIELD_HEADLESS_REDIRECTION))
{
newlength = FIELD_OFFSET(BOOT_OPTIONS, HeadlessRedirection);
newlength += StringLengthW(pHeadlessRedirection,0);
newlength = ALIGN_UP(newlength, ULONG);
}
else
{
newlength = pBootOptions->Length;
}
//Also allocate the memory for a new Boot option struct
pModifiedBootOptions = (PBOOT_OPTIONS)AllocateMemory(newlength);
if(pModifiedBootOptions == NULL)
{
FreeMemory((LPVOID *)&pBootOptions);
return (EXIT_FAILURE);
}
//Fill in the new boot options struct
pModifiedBootOptions->Version = BOOT_OPTIONS_VERSION;
pModifiedBootOptions->Length = newlength;
if((Flag & BOOT_OPTIONS_FIELD_COUNTDOWN))
{
pModifiedBootOptions->Timeout = Timeout;
}
else
{
pModifiedBootOptions->Timeout = pBootOptions->Timeout;
}
//Cannot change the CurrentBootEntryId.So just pass what u got.
pModifiedBootOptions->CurrentBootEntryId = pBootOptions->CurrentBootEntryId;
if((Flag & BOOT_OPTIONS_FIELD_NEXT_BOOT_ENTRY_ID))
{
pModifiedBootOptions->NextBootEntryId = pBootOptions->NextBootEntryId;
}
else
{
pModifiedBootOptions->NextBootEntryId = pBootOptions->NextBootEntryId;
}
if((Flag & BOOT_OPTIONS_FIELD_HEADLESS_REDIRECTION))
{
StringCopy(pModifiedBootOptions->HeadlessRedirection, pBootOptions->HeadlessRedirection, StringLengthW(pBootOptions->HeadlessRedirection,0));
}
else
{
StringCopy(pModifiedBootOptions->HeadlessRedirection, pBootOptions->HeadlessRedirection, StringLengthW(pBootOptions->HeadlessRedirection,0));
}
//Set the boot options in the NVRAM
status = NtSetBootOptions(pModifiedBootOptions, Flag);
if(status != STATUS_SUCCESS)
{
dwExitCode = EXIT_SUCCESS;
if((Flag & BOOT_OPTIONS_FIELD_COUNTDOWN))
{
DISPLAY_MESSAGE(stderr,GetResString(IDS_ERROR_MODIFY_TIMEOUT));
}
if((Flag & BOOT_OPTIONS_FIELD_NEXT_BOOT_ENTRY_ID))
{
DISPLAY_MESSAGE(stderr,GetResString(IDS_ERROR_MODIFY_NEXTBOOTID));
}
if((Flag & BOOT_OPTIONS_FIELD_HEADLESS_REDIRECTION))
{
DISPLAY_MESSAGE(stderr,GetResString(IDS_ERROR_MODIFY_HEADLESS));
}
}
else
{
dwExitCode = EXIT_SUCCESS;
if((Flag & BOOT_OPTIONS_FIELD_COUNTDOWN))
{
ShowMessage(stdout,GetResString(IDS_SUCCESS_MODIFY_TIMEOUT));
}
if((Flag & BOOT_OPTIONS_FIELD_NEXT_BOOT_ENTRY_ID))
{
ShowMessage(stdout,GetResString(IDS_SUCCESS_MODIFY_NEXTBOOTID));
}
if((Flag & BOOT_OPTIONS_FIELD_HEADLESS_REDIRECTION))
{
ShowMessage(stdout,GetResString(IDS_SUCCESS_MODIFY_HEADLESS));
}
}
//free the memory
FreeMemory((LPVOID *) &pModifiedBootOptions);
FreeMemory((LPVOID *) &pBootOptions);
return dwExitCode;
}
DWORD
ConvertBootEntries(PBOOT_ENTRY_LIST NtBootEntries)
/*++
Routine Description :
Convert boot entries read from EFI NVRAM into our internal format.
Arguments :
[ in ] NtBootEntries - The boot entry list given by the enumeration
Return Type : DWORD
--*/
{
PBOOT_ENTRY_LIST bootEntryList;
PBOOT_ENTRY bootEntry;
PBOOT_ENTRY bootEntryCopy;
PMY_BOOT_ENTRY myBootEntry;
PWINDOWS_OS_OPTIONS osOptions;
ULONG length = 0;
DWORD dwErrorCode = EXIT_SUCCESS;
BOOL bNoBreak = TRUE;
bootEntryList = NtBootEntries;
do
{
bootEntry = &bootEntryList->BootEntry;
//
// Calculate the length of our internal structure. This includes
// the base part of MY_BOOT_ENTRY plus the NT BOOT_ENTRY.
//
length = FIELD_OFFSET(MY_BOOT_ENTRY, NtBootEntry) + bootEntry->Length;
//Remember to check for the NULL pointer
myBootEntry = (PMY_BOOT_ENTRY)AllocateMemory(length);
if(myBootEntry == NULL)
{
ShowLastErrorEx( stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return (EXIT_FAILURE);
}
RtlZeroMemory(myBootEntry, length);
//
// Link the new entry into the list.
//
if ( (bootEntry->Attributes & BOOT_ENTRY_ATTRIBUTE_ACTIVE) )
{
InsertTailList( &ActiveUnorderedBootEntries, &myBootEntry->ListEntry );
myBootEntry->ListHead = &ActiveUnorderedBootEntries;
}
else
{
InsertTailList( &InactiveUnorderedBootEntries, &myBootEntry->ListEntry );
myBootEntry->ListHead = &InactiveUnorderedBootEntries;
}
//
// Copy the NT BOOT_ENTRY into the allocated buffer.
//
bootEntryCopy = &myBootEntry->NtBootEntry;
memcpy(bootEntryCopy, bootEntry, bootEntry->Length);
//
// Fill in the base part of the structure.
//
myBootEntry->AllocationEnd = (PUCHAR)myBootEntry + length - 1;
myBootEntry->Id = bootEntry->Id;
//Assign 0 to the Ordered field currently so that
//once the boot order is known, we can assign 1 if this entry is a part of the ordered list.
myBootEntry->Ordered = 0;
myBootEntry->Attributes = bootEntry->Attributes;
myBootEntry->FriendlyName = (PWSTR)ADD_OFFSET(bootEntryCopy, FriendlyNameOffset);
myBootEntry->FriendlyNameLength =((ULONG)StringLengthW(myBootEntry->FriendlyName,0) + 1) * sizeof(WCHAR);
myBootEntry->BootFilePath = (PFILE_PATH)ADD_OFFSET(bootEntryCopy, BootFilePathOffset);
//
// If this is an NT boot entry, capture the NT-specific information in
// the OsOptions.
//
osOptions = (PWINDOWS_OS_OPTIONS)bootEntryCopy->OsOptions;
if ((bootEntryCopy->OsOptionsLength >= FIELD_OFFSET(WINDOWS_OS_OPTIONS, OsLoadOptions)) &&
(strcmp((char *)osOptions->Signature, WINDOWS_OS_OPTIONS_SIGNATURE) == 0))
{
MBE_SET_IS_NT( myBootEntry );
myBootEntry->OsLoadOptions = osOptions->OsLoadOptions;
myBootEntry->OsLoadOptionsLength =((ULONG)StringLengthW(myBootEntry->OsLoadOptions,0) + 1) * sizeof(WCHAR);
myBootEntry->OsFilePath = (PFILE_PATH)ADD_OFFSET(osOptions, OsLoadPathOffset);
}
else
{
//
// Foreign boot entry. Just capture whatever OS options exist.
//
myBootEntry->ForeignOsOptions = bootEntryCopy->OsOptions;
myBootEntry->ForeignOsOptionsLength = bootEntryCopy->OsOptionsLength;
}
//
// Move to the next entry in the enumeration list, if any.
//
if (bootEntryList->NextEntryOffset == 0)
{
bNoBreak = FALSE;
break;
}
bootEntryList = (PBOOT_ENTRY_LIST)ADD_OFFSET(bootEntryList, NextEntryOffset);
} while ( TRUE == bNoBreak );
return dwErrorCode;
} // ConvertBootEntries
DWORD DisplayBootOptions()
/*++
Name : DisplayBootOptions
Synopsis : Display the boot options
Parameters : NONE
Return Type : DWORD
Global Variables: Global Linked lists for storing the boot entries
LIST_ENTRY BootEntries;
--*/
{
DWORD error;
NTSTATUS status;
PBOOT_OPTIONS pBootOptions;
TCHAR szDisplay[MAX_RES_STRING+1] = NULL_STRING;
//Query the boot options
status = BootCfg_QueryBootOptions(&pBootOptions);
if(status != STATUS_SUCCESS)
{
error = RtlNtStatusToDosError( status );
if(pBootOptions)
{
FreeMemory((LPVOID *)&pBootOptions);
}
return EXIT_FAILURE;
}
//Printout the boot options
ShowMessage(stdout,_T("\n"));
ShowMessage(stdout,GetResString(IDS_OUTPUT_IA64A));
ShowMessage(stdout,GetResString(IDS_OUTPUT_IA64B));
ShowMessageEx(stdout, 1, TRUE, GetResString(IDS_OUTPUT_IA64C), pBootOptions->Timeout);
//display default boot entry id
ShowMessageEx(stdout, 1, TRUE, GetResString(IDS_OUTPUT_IA64P), GetDefaultBootEntry());
//Get the CurrentBootEntryId from the actual Id present in the boot options
SecureZeroMemory( szDisplay, SIZE_OF_ARRAY(szDisplay) );
ShowMessageEx(stdout, 1, TRUE, GetResString(IDS_OUTPUT_IA64D), GetCurrentBootEntryID(pBootOptions->CurrentBootEntryId));
ShowMessage(stdout,L"\n");
#if 0
if(StringLengthW(pBootOptions->HeadlessRedirection) == 0)
{
ShowMessage(stdout,GetResString(IDS_OUTPUT_IA64E));
}
else
{
ShowMessageEx(stdout, 1, TRUE, GetResString(IDS_OUTPUT_IA64F), pBootOptions->HeadlessRedirection);
}
#endif //Commenting out the display of the Headless redirection
//as we cannot query the same through API (its Firmware controlled)
if(pBootOptions)
{
FreeMemory((LPVOID *)&pBootOptions);
}
return EXIT_SUCCESS;
}
VOID DisplayBootEntry()
/*++
Routine Description : Display the boot entries (in an order)
Parameters : NONE
Return Type : DWORD
--*/
{
PLIST_ENTRY listEntry;
PMY_BOOT_ENTRY bootEntry;
PWSTR NtFilePath;
//Printout the boot entires
ShowMessage(stdout,GetResString(IDS_OUTPUT_IA64G));
ShowMessage(stdout,GetResString(IDS_OUTPUT_IA64H));
for (listEntry = BootEntries.Flink;listEntry != &BootEntries; listEntry = listEntry->Flink)
{
//Get the boot entry
bootEntry = CONTAINING_RECORD( listEntry, MY_BOOT_ENTRY, ListEntry );
ShowMessageEx(stdout, 1, TRUE, GetResString(IDS_OUTPUT_IA64I), bootEntry->myId);
//friendly name
if(StringLengthW(bootEntry->FriendlyName,0)!=0)
{
ShowMessageEx(stdout, 1, TRUE, GetResString(IDS_OUTPUT_IA64J), bootEntry->FriendlyName);
}
else
{
ShowMessage(stdout,GetResString(IDS_OUTPUT_IA64K));
}
if(MBE_IS_NT(bootEntry))
{
//the OS load options
if(StringLengthW(bootEntry->OsLoadOptions, 0)!=0)
{
ShowMessageEx(stdout, 1, TRUE, GetResString(IDS_OUTPUT_IA64L), bootEntry->OsLoadOptions);
}
else
{
ShowMessage(stdout,GetResString(IDS_OUTPUT_IA64M));
}
//Get the BootFilePath
NtFilePath = GetNtNameForFilePath(bootEntry->BootFilePath);
ShowMessageEx(stdout, 1, TRUE, GetResString(IDS_OUTPUT_IA64N), NtFilePath);
//free the memory
if(NtFilePath)
{
FreeMemory((LPVOID *)&NtFilePath);
}
//Get the OS load path
NtFilePath = GetNtNameForFilePath(bootEntry->OsFilePath);
ShowMessageEx(stdout, 1, TRUE, GetResString(IDS_OUTPUT_IA64O), NtFilePath);
//free the memory
if(NtFilePath)
{
FreeMemory((LPVOID *)&NtFilePath);
}
}
else
{
ShowMessage(stdout,_T("\n"));
}
}
}
DWORD GetCurrentBootEntryID(DWORD Id)
/*++
Routine Description :
Gets the Boot entry ID generated by us from the BootId given by the NVRAM
Arguments :
[ in ] Id - The current boot id (BootId given by the NVRAM)
Return Type : DWORD
Returns EXIT_SUCCESS if successful,
returns EXIT_FAILURE otherwise
--*/
{
PLIST_ENTRY listEntry;
PMY_BOOT_ENTRY bootEntry;
for (listEntry = BootEntries.Flink;listEntry != &BootEntries;listEntry = listEntry->Flink)
{
//Get the boot entry
bootEntry = CONTAINING_RECORD( listEntry, MY_BOOT_ENTRY, ListEntry );
if(bootEntry->Id == Id)
{
return bootEntry->myId;
}
}
return 0;
}
DWORD
ChangeDefaultBootEntry_IA64( IN DWORD argc,
IN LPCTSTR argv[]
)
/*++
Routine Description :
This routine is to change the Default boot entry in the NVRAM
Arguments :
[ in ] argc - Number of command line arguments
[ in ] argv - Array containing command line arguments
Return Type : DWORD
--*/
{
PMY_BOOT_ENTRY mybootEntry;
PLIST_ENTRY listEntry;
NTSTATUS status;
PULONG BootEntryOrder, NewBootEntryOrder;
DWORD dwBootID = 0;
BOOL bDefaultOs = FALSE ;
DWORD dwExitCode = ERROR_SUCCESS;
BOOL bBootIdFound = FALSE;
BOOL bIdFoundInBootOrderList = FALSE;
ULONG length, i, j, defaultId = 0;
DWORD error = 0;
DWORD dwExitcode = 0;
BOOL bUsage = FALSE;
TCMDPARSER2 cmdOptions[3];
PTCMDPARSER2 pcmdOption;
SecureZeroMemory(cmdOptions, SIZE_OF_ARRAY(cmdOptions)*sizeof(TCMDPARSER2) );
//main option
pcmdOption = &cmdOptions[0];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_DEFAULTOS;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bDefaultOs;
//id option
pcmdOption = &cmdOptions[1];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_ID;
pcmdOption->dwFlags = CP_VALUE_MANDATORY | CP2_MANDATORY;
pcmdOption->dwType = CP_TYPE_UNUMERIC;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &dwBootID;
//usage option
pcmdOption = &cmdOptions[2];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_USAGE;
pcmdOption->dwFlags = CP2_USAGE;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bUsage;
if( !DoParseParam2( argc, argv, 0, SIZE_OF_ARRAY(cmdOptions ), cmdOptions, 0 ) )
{
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return (EXIT_FAILURE );
}
//check if usage is specified with more than one option
if( (TRUE == bUsage) && (argc > 3) )
{
ShowMessage(stderr,GetResString(IDS_DEFAULTOS_USAGE));
return ( EXIT_FAILURE );
}
if(bUsage)
{
displayDefaultEntryUsage_IA64();
return EXIT_SUCCESS;
}
dwExitcode = InitializeEFI();
if(EXIT_FAILURE == dwExitcode )
{
return EXIT_FAILURE ;
}
//Check whether the boot entry entered bu the user is a valid boot entry id or not.
for (listEntry = BootEntries.Flink;listEntry != &BootEntries;listEntry = listEntry->Flink)
{
//Get the boot entry
mybootEntry = CONTAINING_RECORD( listEntry, MY_BOOT_ENTRY, ListEntry );
if(mybootEntry->myId == dwBootID)
{
bBootIdFound = TRUE;
//store the default ID
defaultId = mybootEntry->Id;
break;
}
}
if(bBootIdFound == FALSE)
{
//Could not find the BootID specified by the user so output the message and return failure
ShowMessage(stderr,GetResString(IDS_INVALID_BOOTID));
return (EXIT_FAILURE);
}
// Get the system boot order list.
length = 0;
status = NtQueryBootEntryOrder( NULL, &length );
if ( status != STATUS_BUFFER_TOO_SMALL )
{
if ( status == STATUS_SUCCESS )
{
length = 0;
}
else
{
error = RtlNtStatusToDosError( status );
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_DEFAULT_ENTRY),dwBootID);
return (EXIT_FAILURE);
}
}
if ( length != 0 )
{
BootEntryOrder = (PULONG)AllocateMemory( length * sizeof(ULONG) );
if(BootEntryOrder == NULL)
{
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return (EXIT_FAILURE);
}
status = NtQueryBootEntryOrder( BootEntryOrder, &length );
if ( status != STATUS_SUCCESS )
{
error = RtlNtStatusToDosError( status );
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_DEFAULT_ENTRY),dwBootID);
FreeMemory((LPVOID *)&BootEntryOrder);
return EXIT_FAILURE;
}
}
//Check if the boot id entered by the user is a part of the Boot entry order.
//If not for the time being do not make it the default.
for(i=0;i<length;i++)
{
if(*(BootEntryOrder+i) == defaultId)
{
bIdFoundInBootOrderList = TRUE;
break;
}
}
if(bIdFoundInBootOrderList == FALSE)
{
FreeMemory((LPVOID *)&BootEntryOrder);
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_DEFAULT_ENTRY),dwBootID);
return (EXIT_FAILURE);
}
//Allocate memory for storing the new boot entry order.
NewBootEntryOrder = (PULONG)AllocateMemory((length) * sizeof(ULONG));
if(NewBootEntryOrder == NULL)
{
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
FreeMemory((LPVOID *)&BootEntryOrder);
return (EXIT_FAILURE);
}
*NewBootEntryOrder = defaultId;
j=0;
for(i=0;i<length;i++)
{
if(*(BootEntryOrder+i) == defaultId)
{
continue;
}
*(NewBootEntryOrder+(j+1)) = *(BootEntryOrder+i);
j++;
}
status = NtSetBootEntryOrder(NewBootEntryOrder, length);
if ( status != STATUS_SUCCESS )
{
error = RtlNtStatusToDosError( status );
dwExitCode = error;
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_DEFAULT_ENTRY),dwBootID);
}
else
{
ShowMessageEx(stdout, 1, TRUE, GetResString(IDS_SUCCESS_DEFAULT_ENTRY),dwBootID);
}
//free the memory
FreeMemory((LPVOID *)&NewBootEntryOrder);
FreeMemory((LPVOID *)&BootEntryOrder);
return dwExitCode;
}
DWORD
ProcessDebugSwitch_IA64( IN DWORD argc,
IN LPCTSTR argv[]
)
/*++
Routine Description :
Allows the user to add the OS load options specifed
as a debug string at the cmdline to the boot
Arguments :
[ in ] argc - Number of command line arguments
[ in ] argv - Array containing command line arguments
Return Type : DWORD
Returns EXIT_SUCCESS if successful,
returns EXIT_FAILURE otherwise.
--*/
{
BOOL bUsage = FALSE ;
DWORD dwBootID = 0;
BOOL bBootIdFound = FALSE;
DWORD dwExitCode = ERROR_SUCCESS;
PMY_BOOT_ENTRY mybootEntry;
PLIST_ENTRY listEntry;
PBOOT_ENTRY bootEntry;
TCHAR szPort[MAX_RES_STRING+1] = NULL_STRING ;
TCHAR szBaudRate[MAX_RES_STRING+1] = NULL_STRING ;
TCHAR szDebug[MAX_RES_STRING+1] = NULL_STRING ;
BOOL bDebug = FALSE ;
PWINDOWS_OS_OPTIONS pWindowsOptions = NULL ;
TCHAR szOsLoadOptions[MAX_RES_STRING+1] = NULL_STRING ;
TCHAR szTemp[MAX_RES_STRING+1] = NULL_STRING ;
TCHAR szTmpBuffer[MAX_RES_STRING+1] = NULL_STRING ;
DWORD dwExitcode = 0 ;
TCMDPARSER2 cmdOptions[6];
PTCMDPARSER2 pcmdOption;
SecureZeroMemory(cmdOptions, SIZE_OF_ARRAY(cmdOptions)*sizeof(TCMDPARSER2) );
//main option
pcmdOption = &cmdOptions[0];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_DEBUG;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bDebug;
//usage
pcmdOption = &cmdOptions[1];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_USAGE;
pcmdOption->dwFlags = CP2_USAGE;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bUsage;
//id option
pcmdOption = &cmdOptions[2];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_ID;
pcmdOption->dwFlags = CP_VALUE_MANDATORY | CP2_MANDATORY;
pcmdOption->dwType = CP_TYPE_UNUMERIC;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &dwBootID;
//port option
pcmdOption = &cmdOptions[3];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_PORT;
pcmdOption->dwFlags = CP_VALUE_MANDATORY | CP2_MODE_VALUES | CP2_VALUE_TRIMINPUT | CP2_VALUE_NONULL;;
pcmdOption->dwType = CP_TYPE_TEXT;
pcmdOption->dwCount = 1;
pcmdOption->pValue = szPort;
pcmdOption->pwszValues = COM_PORT_RANGE;
pcmdOption->dwLength= MAX_STRING_LENGTH;
//baud option
pcmdOption = &cmdOptions[4];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_BAUD;
pcmdOption->dwFlags = CP_VALUE_MANDATORY | CP2_MODE_VALUES | CP2_VALUE_TRIMINPUT | CP2_VALUE_NONULL;;
pcmdOption->dwType = CP_TYPE_TEXT;
pcmdOption->dwCount = 1;
pcmdOption->pValue = szBaudRate;
pcmdOption->pwszValues = BAUD_RATE_VALUES_DEBUG;
pcmdOption->dwLength= MAX_STRING_LENGTH;
//default on/off option
pcmdOption = &cmdOptions[5];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_DEFAULT;
pcmdOption->dwFlags = CP2_DEFAULT | CP2_MANDATORY | CP2_VALUE_TRIMINPUT | CP2_VALUE_NONULL;;
pcmdOption->dwType = CP_TYPE_TEXT;
pcmdOption->dwCount = 1;
pcmdOption->pValue = szDebug;
pcmdOption->dwLength= MAX_STRING_LENGTH;
// Parsing the copy option switches
if ( !DoParseParam2( argc, argv, 0, SIZE_OF_ARRAY(cmdOptions ), cmdOptions, 0 ) )
{
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return (EXIT_FAILURE);
}
//check if usage is specified with more than one option
if( (TRUE == bUsage) && (argc > 3) )
{
ShowMessage(stderr,GetResString(IDS_INVALID_SYNTAX_DEBUG));
return ( EXIT_FAILURE );
}
// Displaying query usage if user specified -? with -query option
if( bUsage )
{
displayDebugUsage_IA64();
return (ERROR_SUCCESS);
}
dwExitcode = InitializeEFI();
if(EXIT_FAILURE == dwExitcode )
{
return EXIT_FAILURE ;
}
//Trim any leading or trailing spaces
if(StringLengthW(szDebug, 0)!=0)
{
TrimString(szDebug, TRIM_ALL);
}
if( !( ( StringCompare(szDebug,VALUE_ON,TRUE,0)== 0) || (StringCompare(szDebug,VALUE_OFF,TRUE,0)== 0 ) ||(StringCompare(szDebug,EDIT_STRING,TRUE,0)== 0) ) )
{
ShowMessage(stderr,GetResString(IDS_INVALID_SYNTAX_DEBUG));
return EXIT_FAILURE;
}
//Query the boot entries till u get the BootID specified by the user
for (listEntry = BootEntries.Flink;listEntry != &BootEntries; listEntry = listEntry->Flink)
{
//Get the boot entry
mybootEntry = CONTAINING_RECORD( listEntry, MY_BOOT_ENTRY, ListEntry );
if(mybootEntry->myId == dwBootID)
{
bBootIdFound = TRUE;
bootEntry = &mybootEntry->NtBootEntry;
//Check whether the bootEntry is a Windows one or not.
//The OS load options can be added only to a Windows boot entry.
if(!IsBootEntryWindows(bootEntry))
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_OSOPTIONS),dwBootID);
dwExitCode = EXIT_FAILURE;
break;
}
//Change the OS load options. Pass NULL to friendly name as we are not changing the same
//szRawString is the Os load options specified by the user
pWindowsOptions = (PWINDOWS_OS_OPTIONS)bootEntry->OsOptions;
if(StringLengthW(pWindowsOptions->OsLoadOptions, 0) > MAX_RES_STRING)
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_STRING_LENGTH1),MAX_RES_STRING);
return EXIT_FAILURE ;
}
// copy the existing OS Loadoptions into a string.
StringCopy(szOsLoadOptions,pWindowsOptions->OsLoadOptions, SIZE_OF_ARRAY(szOsLoadOptions));
//check if the user has entered On option
if( StringCompare(szDebug,VALUE_ON,TRUE,0)== 0)
{
//display an error message
if ( (FindString(szOsLoadOptions,DEBUG_SWITCH, 0) != 0 )&& (StringLengthW(szPort, 0)==0) &&(StringLengthW(szBaudRate, 0)==0) )
{
ShowMessage(stderr,GetResString(IDS_DUPL_DEBUG));
dwExitCode = EXIT_FAILURE;
break;
}
//display a error message and exit if the 1394 port is already present.
if(FindString(szOsLoadOptions,DEBUGPORT_1394, 0) != 0 )
{
ShowMessage(stderr,GetResString(IDS_1394_ALREADY_PRESENT));
dwExitCode = EXIT_FAILURE;
break;
}
//
//display an duplicate entry error message if substring is already present.
//
if(StringLengthW(szBaudRate, 0)==0)
{
if ( GetSubString(szOsLoadOptions,TOKEN_DEBUGPORT,szTemp) == EXIT_SUCCESS )
{
ShowMessage(stderr,GetResString(IDS_DUPLICATE_ENTRY));
return EXIT_FAILURE ;
}
}
if(StringLengthW(szTemp, 0)!=0)
{
ShowMessage(stderr,GetResString(IDS_DUPLICATE_ENTRY));
dwExitCode = EXIT_FAILURE;
break;
}
//check if the Os load options already contains
// debug switch
if(FindString(szOsLoadOptions,DEBUG_SWITCH, 0) == 0 )
{
if(StringLengthW(szOsLoadOptions, 0)!=0)
{
StringCopy(szTmpBuffer,TOKEN_EMPTYSPACE, SIZE_OF_ARRAY(szTmpBuffer));
StringConcat(szTmpBuffer,DEBUG_SWITCH, SIZE_OF_ARRAY(szOsLoadOptions) );
}
else
{
StringCopy(szTmpBuffer,DEBUG_SWITCH, SIZE_OF_ARRAY(szOsLoadOptions) );
}
}
if(StringLengthW(szPort, 0)!= 0)
{
if(StringLengthW(szTmpBuffer, 0)==0)
{
StringCopy(szTmpBuffer,TOKEN_EMPTYSPACE, SIZE_OF_ARRAY(szTmpBuffer));
}
else
{
StringConcat(szTmpBuffer,TOKEN_EMPTYSPACE, SIZE_OF_ARRAY(szTmpBuffer));
}
StringConcat(szTmpBuffer,TOKEN_DEBUGPORT, SIZE_OF_ARRAY(szTmpBuffer)) ;
StringConcat(szTmpBuffer,TOKEN_EQUAL, SIZE_OF_ARRAY(szTmpBuffer)) ;
CharUpper(szPort);
StringConcat(szTmpBuffer,szPort, SIZE_OF_ARRAY(szTmpBuffer));
}
//Check if the OS Load Options contains the baud rate already specified.
if(StringLengthW(szBaudRate, 0)!=0)
{
StringCopy(szTemp,NULL_STRING, SIZE_OF_ARRAY(szTemp) );
GetBaudRateVal(szOsLoadOptions,szTemp) ;
if(StringLengthW(szTemp, 0)!=0)
{
ShowMessage(stderr,GetResString(IDS_DUPLICATE_BAUD_RATE));
dwExitCode = EXIT_FAILURE;
break;
}
else
{
if(StringLengthW(szTmpBuffer, 0)==0)
{
StringCopy(szTmpBuffer,TOKEN_EMPTYSPACE, SIZE_OF_ARRAY(szTmpBuffer));
}
else
{
StringConcat(szTmpBuffer,TOKEN_EMPTYSPACE, SIZE_OF_ARRAY(szTmpBuffer));
}
StringConcat(szTmpBuffer,BAUD_RATE, SIZE_OF_ARRAY(szTmpBuffer));
StringConcat(szTmpBuffer,TOKEN_EQUAL, SIZE_OF_ARRAY(szTmpBuffer));
StringConcat(szTmpBuffer,szBaudRate, SIZE_OF_ARRAY(szTmpBuffer));
}
}
}
//check if the user has entered OFF option
if( StringCompare(szDebug,VALUE_OFF,TRUE,0)== 0)
{
// If the user enters either com port or baud rate then display error message and exit.
if ((StringLengthW(szPort, 0)!=0) ||(StringLengthW(szBaudRate, 0)!=0))
{
ShowMessage(stderr,GetResString(IDS_INVALID_SYNTAX_DEBUG));
dwExitCode = EXIT_FAILURE;
break;
}
// if debug port is absent print message and exit.
if (FindString(szOsLoadOptions,DEBUG_SWITCH, 0) == 0 )
{
ShowMessage(stderr,GetResString(IDS_DEBUG_ABSENT));
dwExitCode = EXIT_FAILURE;
break;
}
//remove the debug switch from the OSLoad Options
removeSubString(szOsLoadOptions,DEBUG_SWITCH);
if(FindString(szOsLoadOptions,DEBUGPORT_1394, 0) != 0 )
{
ShowMessage(stderr,GetResString(IDS_ERROR_1394_REMOVE));
dwExitCode = EXIT_FAILURE;
break;
}
GetSubString(szOsLoadOptions, TOKEN_DEBUGPORT,szTemp);
if(StringLengthW(szTemp, 0)!=0)
{
// remove the /debugport=comport switch if it is present from the Boot Entry
removeSubString(szOsLoadOptions,szTemp);
}
StringCopy(szTemp , NULL_STRING, SIZE_OF_ARRAY(szTemp) );
//remove the baud rate switch if it is present.
GetBaudRateVal(szOsLoadOptions,szTemp) ;
// if the OSLoadOptions contains baudrate then delete it.
if (StringLengthW(szTemp, 0 )!= 0)
{
removeSubString(szOsLoadOptions,szTemp);
}
}
//if the user selected the edit option .
if( StringCompare(szDebug,EDIT_STRING,TRUE,0)== 0)
{
//check if the debug switch is present in the Osload options else display error message.
if (FindString(szOsLoadOptions,DEBUG_SWITCH, 0) == 0 )
{
ShowMessage(stderr,GetResString(IDS_DEBUG_ABSENT));
dwExitCode = EXIT_FAILURE;
break;
}
if( FindString(szOsLoadOptions,DEBUGPORT_1394, 0) != 0 )
{
ShowMessage(stderr,GetResString(IDS_ERROR_EDIT_1394_SWITCH));
dwExitCode = EXIT_FAILURE;
break;
}
//check if the user enters COM port or baud rate else display error message.
if((StringLengthW(szPort, 0)==0)&&(StringLengthW(szBaudRate, 0)==0))
{
ShowMessage(stderr,GetResString(IDS_INVALID_SYNTAX_DEBUG));
dwExitCode = EXIT_FAILURE;
break;
}
if( StringLengthW(szPort, 0)!=0 )
{
if ( GetSubString(szOsLoadOptions,TOKEN_DEBUGPORT,szTemp) == EXIT_FAILURE)
{
ShowMessage(stderr,GetResString(IDS_NO_DEBUGPORT));
return EXIT_FAILURE ;
}
if(StringLengthW(szTemp, 0)!=0)
{
//remove the existing entry from the OsLoadOptions String.
removeSubString(szOsLoadOptions,szTemp);
}
//Add the port entry specified by user into the OsLoadOptions String.
if(StringLengthW(szTmpBuffer, 0)==0)
{
StringCopy(szTmpBuffer,TOKEN_EMPTYSPACE, SIZE_OF_ARRAY(szTmpBuffer));
}
else
{
StringConcat(szTmpBuffer,TOKEN_EMPTYSPACE, SIZE_OF_ARRAY(szTmpBuffer));
}
StringConcat(szTmpBuffer,TOKEN_DEBUGPORT, SIZE_OF_ARRAY(szTmpBuffer)) ;
StringConcat(szTmpBuffer,TOKEN_EQUAL, SIZE_OF_ARRAY(szTmpBuffer)) ;
CharUpper(szPort);
StringConcat(szTmpBuffer,szPort, SIZE_OF_ARRAY(szTmpBuffer));
}
//Check if the OS Load Options contains the baud rate already specified.
if(StringLengthW(szBaudRate, 0)!=0)
{
StringCopy(szTemp,NULL_STRING, SIZE_OF_ARRAY(szTemp));
GetBaudRateVal(szOsLoadOptions,szTemp) ;
if(StringLengthW(szTemp, 0)!=0)
{
removeSubString(szOsLoadOptions,szTemp);
}
//add the baud rate value to boot entry
if(StringLengthW(szTmpBuffer, 0) == 0)
{
StringCopy(szTmpBuffer,TOKEN_EMPTYSPACE, SIZE_OF_ARRAY(szTmpBuffer));
}
else
{
StringConcat(szTmpBuffer,TOKEN_EMPTYSPACE, SIZE_OF_ARRAY(szTmpBuffer));
}
StringConcat(szTmpBuffer,BAUD_RATE, SIZE_OF_ARRAY(szTmpBuffer));
StringConcat(szTmpBuffer,TOKEN_EQUAL, SIZE_OF_ARRAY(szTmpBuffer));
StringConcat(szTmpBuffer,szBaudRate, SIZE_OF_ARRAY(szTmpBuffer));
}
}
//display error message if Os Load options is more than 254
// characters.
if(StringLengthW(szOsLoadOptions, 0) + StringLengthW(szTmpBuffer,0)> MAX_RES_STRING)
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_STRING_LENGTH1),MAX_RES_STRING);
return EXIT_FAILURE ;
}
else
{
StringConcat(szOsLoadOptions,szTmpBuffer, SIZE_OF_ARRAY(szOsLoadOptions));
}
// modify the Boot Entry with the modified OsLoad Options.
dwExitCode = ChangeBootEntry(bootEntry, NULL, szOsLoadOptions);
if(dwExitCode == ERROR_SUCCESS)
{
ShowMessageEx(stdout,1,TRUE, GetResString(IDS_SUCCESS_CHANGE_OSOPTIONS),dwBootID);
}
else
{
ShowMessageEx(stderr,1,TRUE, GetResString(IDS_ERROR_CHANGE_OSOPTIONS),dwBootID);
}
break;
}
}
if(FALSE == bBootIdFound )
{
//Could not find the BootID specified by the user so output the message and return failure
ShowMessage(stderr,GetResString(IDS_INVALID_BOOTID));
dwExitCode = EXIT_FAILURE;
}
//Remember to free memory allocated for the linked lists
Freelist();
return (dwExitCode);
}
VOID
GetComPortType_IA64( IN LPTSTR szString,
IN LPTSTR szTemp
)
/*++
Routine Description:
Get the Type of Com Port present in Boot Entry
Arguments :
[ in ] szString : The String which is to be searched.
[ in ] szTemp : String which will get the com port type
Return Type : VOID
--*/
{
if(FindString(szString,PORT_COM1A, 0)!=0)
{
StringCopy(szTemp,PORT_COM1A, MAX_RES_STRING);
}
else if(FindString(szString,PORT_COM2A,0)!=0)
{
StringCopy(szTemp,PORT_COM2A, MAX_RES_STRING);
}
else if(FindString(szString,PORT_COM3A,0)!=0)
{
StringCopy(szTemp,PORT_COM3A, MAX_RES_STRING);
}
else if(FindString(szString,PORT_COM4A,0)!=0)
{
StringCopy(szTemp,PORT_COM4A, MAX_RES_STRING);
}
else if(FindString(szString,PORT_1394A,0)!=0)
{
StringCopy(szTemp,PORT_1394A, MAX_RES_STRING);
}
}
DWORD
ProcessEmsSwitch_IA64( IN DWORD argc,
IN LPCTSTR argv[]
)
/*++
Routine Description :
Which process the ems switch.
Arguments :
[ in ] argc - Number of command line arguments
[ in ] argv - Array containing command line arguments
Return Type : DWORD
--*/
{
PMY_BOOT_ENTRY mybootEntry;
PLIST_ENTRY listEntry;
PBOOT_ENTRY bootEntry;
PWINDOWS_OS_OPTIONS pWindowsOptions = NULL ;
BOOL bUsage = FALSE ;
DWORD dwBootID = 0;
BOOL bBootIdFound = FALSE;
DWORD dwExitCode = ERROR_SUCCESS;
TCHAR szEms[MAX_STRING_LENGTH+1] = NULL_STRING ;
BOOL bEms = FALSE ;
TCHAR szOsLoadOptions[MAX_RES_STRING+1] = NULL_STRING ;
TCHAR szTmpBuffer[MAX_RES_STRING+1] = NULL_STRING ;
DWORD dwExitcode = 0 ;
TCMDPARSER2 cmdOptions[4];
PTCMDPARSER2 pcmdOption;
SecureZeroMemory(cmdOptions, SIZE_OF_ARRAY(cmdOptions)*sizeof(TCMDPARSER2) );
//main option
pcmdOption = &cmdOptions[0];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_EMS;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bEms;
//usage
pcmdOption = &cmdOptions[1];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_USAGE;
pcmdOption->dwFlags = CP2_USAGE;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bUsage;
//id option
pcmdOption = &cmdOptions[2];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_ID;
pcmdOption->dwFlags = CP_VALUE_MANDATORY | CP2_MANDATORY;
pcmdOption->dwType = CP_TYPE_UNUMERIC;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &dwBootID;
//default on/off option
pcmdOption = &cmdOptions[3];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_DEFAULT;
pcmdOption->dwFlags = CP2_DEFAULT | CP2_MANDATORY | CP2_VALUE_TRIMINPUT | CP2_VALUE_NONULL;;
pcmdOption->dwType = CP_TYPE_TEXT;
pcmdOption->dwCount = 1;
pcmdOption->pValue = szEms;
pcmdOption->dwLength= MAX_STRING_LENGTH;
// Parsing the ems option switches
if ( !DoParseParam2( argc, argv, 0, SIZE_OF_ARRAY(cmdOptions ), cmdOptions, 0 ) )
{
ShowLastErrorEx( stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return (EXIT_FAILURE);
}
//check if usage is specified with more than one option
if( (TRUE == bUsage) && (argc > 3) )
{
ShowMessage(stderr,GetResString(IDS_INVALID_SYNTAX_EMS));
return ( EXIT_FAILURE );
}
// Displaying query usage if user specified -? with -query option
if( bUsage )
{
displayEmsUsage_IA64();
return (EXIT_SUCCESS);
}
dwExitcode = InitializeEFI();
if(EXIT_FAILURE == dwExitcode )
{
return EXIT_FAILURE ;
}
//display error message if the user enters any other string other that on/off.
if( !((StringCompare(szEms,VALUE_ON,TRUE,0)== 0) || (StringCompare(szEms,VALUE_OFF,TRUE,0)== 0)))
{
ShowMessage(stderr,GetResString(IDS_INVALID_SYNTAX_EMS));
return EXIT_FAILURE ;
}
//Query the boot entries till u get the BootID specified by the user
for (listEntry = BootEntries.Flink;listEntry != &BootEntries;listEntry = listEntry->Flink)
{
//Get the boot entry
mybootEntry = CONTAINING_RECORD( listEntry, MY_BOOT_ENTRY, ListEntry );
if(mybootEntry->myId == dwBootID)
{
bBootIdFound = TRUE;
bootEntry = &mybootEntry->NtBootEntry;
//Check whether the bootEntry is a Windows one or not.
//The OS load options can be added only to a Windows boot entry.
if(!IsBootEntryWindows(bootEntry))
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_OSOPTIONS),dwBootID);
dwExitCode = EXIT_FAILURE;
break;
}
pWindowsOptions = (PWINDOWS_OS_OPTIONS)bootEntry->OsOptions;
if(StringLengthW(pWindowsOptions->OsLoadOptions, 0) > MAX_RES_STRING)
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_STRING_LENGTH1),MAX_RES_STRING);
return EXIT_FAILURE ;
}
// copy the existing OS Loadoptions into a string.
StringCopy(szOsLoadOptions,pWindowsOptions->OsLoadOptions, SIZE_OF_ARRAY(szOsLoadOptions));
//check if the user has entered On option
if( StringCompare(szEms,VALUE_ON,TRUE,0)== 0)
{
if (FindString(szOsLoadOptions,REDIRECT_SWITCH, 0) != 0 )
{
ShowMessage(stderr,GetResString(IDS_DUPL_REDIRECT));
dwExitCode = EXIT_FAILURE;
break;
}
// add the redirect switch to the OS Load Options string.
if( StringLength(szOsLoadOptions,0) != 0 )
{
StringCopy(szTmpBuffer,TOKEN_EMPTYSPACE, SIZE_OF_ARRAY(szTmpBuffer));
StringConcat(szTmpBuffer,REDIRECT_SWITCH, SIZE_OF_ARRAY(szTmpBuffer) );
}
else
{
StringCopy(szTmpBuffer,REDIRECT_SWITCH, SIZE_OF_ARRAY(szTmpBuffer) );
}
}
//check if the user has entered OFF option
if( StringCompare(szEms,VALUE_OFF,TRUE,0)== 0)
{
// If the user enters either com port or baud rate then display error message and exit.
if (FindString(szOsLoadOptions,REDIRECT_SWITCH, 0) == 0 )
{
ShowMessage(stderr,GetResString(IDS_REDIRECT_ABSENT));
dwExitCode = EXIT_FAILURE;
break;
}
//remove the debug switch from the OSLoad Options
removeSubString(szOsLoadOptions,REDIRECT_SWITCH);
}
//display error message if Os Load options is more than 254
// characters.
if(StringLengthW(szOsLoadOptions, 0)+StringLength(szTmpBuffer,0) > MAX_RES_STRING)
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_STRING_LENGTH1),MAX_RES_STRING);
return EXIT_FAILURE ;
}
else
{
StringConcat( szOsLoadOptions, szTmpBuffer, SIZE_OF_ARRAY(szOsLoadOptions) );
}
// modify the Boot Entry with the modified OsLoad Options.
dwExitCode = ChangeBootEntry(bootEntry, NULL, szOsLoadOptions);
if(dwExitCode == ERROR_SUCCESS)
{
ShowMessageEx(stdout, 1, TRUE, GetResString(IDS_SUCCESS_CHANGE_OSOPTIONS),dwBootID);
}
else
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_CHANGE_OSOPTIONS),dwBootID);
}
break;
}
}
if(bBootIdFound == FALSE)
{
//Could not find the BootID specified by the user so output the message and return failure
ShowMessage(stderr,GetResString(IDS_INVALID_BOOTID));
dwExitCode = EXIT_FAILURE;
}
//free the global linked lists
Freelist();
return (dwExitCode);
}
DWORD
ProcessAddSwSwitch_IA64( IN DWORD argc,
IN LPCTSTR argv[]
)
/*++
Routine Description :
Which implements the Addsw switch.
Arguments :
[ in ] argc - Number of command line arguments
[ in ] argv - Array containing command line arguments
Return Type : DWORD
Returns EXIT_SUCCESS if it is successful,
return EXIT_FAILURE otherwise.
--*/
{
BOOL bUsage = FALSE ;
BOOL bAddSw = FALSE ;
DWORD dwBootID = 0;
BOOL bBootIdFound = FALSE;
DWORD dwExitCode = ERROR_SUCCESS;
PMY_BOOT_ENTRY mybootEntry;
PLIST_ENTRY listEntry;
PBOOT_ENTRY bootEntry;
BOOL bBaseVideo = FALSE ;
BOOL bNoGui = FALSE ;
BOOL bSos = FALSE ;
DWORD dwMaxmem = 0 ;
PWINDOWS_OS_OPTIONS pWindowsOptions = NULL ;
TCHAR szOsLoadOptions[MAX_RES_STRING+1] = NULL_STRING ;
TCHAR szTmpBuffer[MAX_RES_STRING+1] = NULL_STRING ;
TCHAR szMaxmem[MAX_RES_STRING+1] = NULL_STRING ;
DWORD dwExitcode = 0 ;
TCMDPARSER2 cmdOptions[7];
PTCMDPARSER2 pcmdOption;
SecureZeroMemory(cmdOptions, SIZE_OF_ARRAY(cmdOptions)*sizeof(TCMDPARSER2) );
//main option
pcmdOption = &cmdOptions[0];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_ADDSW;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bAddSw;
// usage
pcmdOption = &cmdOptions[1];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_USAGE;
pcmdOption->dwFlags = CP2_USAGE;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bUsage;
//id option
pcmdOption = &cmdOptions[2];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_ID;
pcmdOption->dwFlags = CP_VALUE_MANDATORY | CP2_MANDATORY;
pcmdOption->dwType = CP_TYPE_UNUMERIC;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &dwBootID;
//maxmem option
pcmdOption = &cmdOptions[3];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_MAXMEM;
pcmdOption->dwFlags = CP_VALUE_MANDATORY;
pcmdOption->dwType = CP_TYPE_UNUMERIC;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &dwMaxmem;
//basvideo option
pcmdOption = &cmdOptions[4];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_BASEVIDEO;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bBaseVideo;
//nogui option
pcmdOption = &cmdOptions[5];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_NOGUIBOOT;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bNoGui;
//nogui option
pcmdOption = &cmdOptions[6];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_SOS;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bSos;
// Parsing the copy option switches
if ( !DoParseParam2( argc, argv, 0, SIZE_OF_ARRAY(cmdOptions ), cmdOptions, 0 ) )
{
ShowLastErrorEx( stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return (EXIT_FAILURE);
}
//check if usage is specified with more than one option
if( (TRUE == bUsage) && (argc > 3) )
{
ShowMessage(stderr,GetResString(IDS_INVALID_SYNTAX_ADDSW));
return ( EXIT_FAILURE );
}
// Displaying query usage if user specified -? with -query option
if( bUsage )
{
displayAddSwUsage_IA64();
return (EXIT_SUCCESS);
}
dwExitcode = InitializeEFI();
if(EXIT_FAILURE == dwExitcode )
{
return EXIT_FAILURE ;
}
if((0==dwMaxmem)&&(cmdOptions[3].dwActuals!=0))
{
ShowMessage(stderr,GetResString(IDS_ERROR_MAXMEM_VALUES));
return EXIT_FAILURE ;
}
//display an error mesage if none of the options are specified.
if((!bSos)&&(!bBaseVideo)&&(!bNoGui)&&(dwMaxmem==0))
{
ShowMessage(stderr,GetResString(IDS_INVALID_SYNTAX_ADDSW));
return EXIT_FAILURE ;
}
//Query the boot entries till u get the BootID specified by the user
for (listEntry = BootEntries.Flink;listEntry != &BootEntries; listEntry = listEntry->Flink)
{
//Get the boot entry
mybootEntry = CONTAINING_RECORD( listEntry, MY_BOOT_ENTRY, ListEntry );
if(mybootEntry->myId == dwBootID)
{
bBootIdFound = TRUE;
bootEntry = &mybootEntry->NtBootEntry;
//Check whether the bootEntry is a Windows one or not.
//The OS load options can be added only to a Windows boot entry.
if(!IsBootEntryWindows(bootEntry))
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_OSOPTIONS),dwBootID);
dwExitCode = EXIT_FAILURE;
break;
}
pWindowsOptions = (PWINDOWS_OS_OPTIONS)bootEntry->OsOptions;
if(StringLengthW(pWindowsOptions->OsLoadOptions,0) > MAX_RES_STRING)
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_STRING_LENGTH1),MAX_RES_STRING);
return EXIT_FAILURE ;
}
// copy the existing OS Loadoptions into a string.
StringCopy(szOsLoadOptions,pWindowsOptions->OsLoadOptions, SIZE_OF_ARRAY(szOsLoadOptions));
//check if the user has entered -basevideo option
if(bBaseVideo)
{
if (FindString(szOsLoadOptions,BASEVIDEO_VALUE, 0) != 0 )
{
ShowMessage(stderr,GetResString(IDS_DUPL_BASEVIDEO_SWITCH));
dwExitCode = EXIT_FAILURE;
break;
}
else
{
StringCopy(szTmpBuffer,BASEVIDEO_VALUE, SIZE_OF_ARRAY(szTmpBuffer) );
}
}
if(bSos)
{
if (FindString(szOsLoadOptions,SOS_VALUE, 0) != 0 )
{
ShowMessage(stderr,GetResString(IDS_DUPL_SOS_SWITCH));
dwExitCode = EXIT_FAILURE;
break;
}
else
{
// add the sos switch to the OS Load Options string.
if(StringLengthW(szTmpBuffer, 0) != 0)
{
StringConcat(szTmpBuffer,TOKEN_EMPTYSPACE, SIZE_OF_ARRAY(szTmpBuffer));
StringConcat(szTmpBuffer,SOS_VALUE, SIZE_OF_ARRAY(szTmpBuffer));
}
else
{
StringCopy(szTmpBuffer,SOS_VALUE, SIZE_OF_ARRAY(szTmpBuffer));
}
}
}
if(bNoGui)
{
if (FindString(szOsLoadOptions,NOGUI_VALUE, 0) != 0 )
{
ShowMessage(stderr,GetResString(IDS_DUPL_NOGUI_SWITCH));
dwExitCode = EXIT_FAILURE;
break;
}
else
{
// add the no gui switch to the OS Load Options string.
if(StringLengthW(szTmpBuffer, 0) != 0)
{
StringConcat(szTmpBuffer,TOKEN_EMPTYSPACE, SIZE_OF_ARRAY(szTmpBuffer));
StringConcat(szTmpBuffer,NOGUI_VALUE, SIZE_OF_ARRAY(szTmpBuffer));
}
else
{
StringCopy(szTmpBuffer,NOGUI_VALUE, SIZE_OF_ARRAY(szTmpBuffer));
}
}
}
if(dwMaxmem!=0)
{
// check if the maxmem value is in the valid range.
if( (dwMaxmem < 32) )
{
ShowMessage(stderr,GetResString(IDS_ERROR_MAXMEM_VALUES));
dwExitCode = EXIT_FAILURE;
break;
}
if (FindString(szOsLoadOptions,MAXMEM_VALUE1, 0) != 0 )
{
ShowMessage(stderr,GetResString(IDS_DUPL_MAXMEM_SWITCH));
dwExitCode = EXIT_FAILURE;
break;
}
else
{
// add the maxmem switch to the OS Load Options string.
if(StringLengthW(szTmpBuffer, 0) != 0)
{
StringConcat(szTmpBuffer,TOKEN_EMPTYSPACE, SIZE_OF_ARRAY(szTmpBuffer));
StringConcat(szTmpBuffer,MAXMEM_VALUE1, SIZE_OF_ARRAY(szTmpBuffer));
StringConcat(szTmpBuffer,TOKEN_EQUAL, SIZE_OF_ARRAY(szTmpBuffer));
_ltow(dwMaxmem,szMaxmem,10);
StringConcat(szTmpBuffer,szMaxmem, SIZE_OF_ARRAY(szTmpBuffer));
}
else
{
StringCopy(szTmpBuffer,MAXMEM_VALUE1, SIZE_OF_ARRAY(szTmpBuffer));
StringConcat(szTmpBuffer,TOKEN_EQUAL, SIZE_OF_ARRAY(szTmpBuffer));
_ltow(dwMaxmem,szMaxmem,10);
StringConcat(szTmpBuffer,szMaxmem, SIZE_OF_ARRAY(szTmpBuffer));
}
}
}
//display error message if Os Load options is more than 254
// characters.
if(StringLengthW(szOsLoadOptions, 0)+StringLength(szTmpBuffer,0) + StringLength(TOKEN_EMPTYSPACE,0)> MAX_RES_STRING)
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_STRING_LENGTH1),MAX_RES_STRING);
return EXIT_FAILURE ;
}
else
{
if( StringLength(szOsLoadOptions,0) != 0 )
{
StringConcat(szOsLoadOptions,TOKEN_EMPTYSPACE, SIZE_OF_ARRAY(szOsLoadOptions));
StringConcat(szOsLoadOptions,szTmpBuffer,SIZE_OF_ARRAY(szOsLoadOptions));
}
else
{
StringCopy(szOsLoadOptions,szTmpBuffer,SIZE_OF_ARRAY(szOsLoadOptions));
}
}
//Change the OS load options. Pass NULL to friendly name as we are not changing the same
//szRawString is the Os load options specified by the user
dwExitCode = ChangeBootEntry(bootEntry, NULL, szOsLoadOptions);
if(dwExitCode == ERROR_SUCCESS)
{
ShowMessageEx(stdout, 1, TRUE, GetResString(IDS_SUCCESS_OSOPTIONS),dwBootID);
}
else
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_OSOPTIONS),dwBootID);
}
break;
}
}
if(bBootIdFound == FALSE)
{
//Could not find the BootID specified by the user so output the message and return failure
ShowMessage(stderr,GetResString(IDS_INVALID_BOOTID));
dwExitCode = EXIT_FAILURE;
}
//Remember to free memory allocated for the linked lists
Freelist();
return (dwExitCode);
}
DWORD
ProcessRmSwSwitch_IA64( IN DWORD argc,
IN LPCTSTR argv[]
)
/*++
Routine Description :
Process the rmsw switch
Arguments :
[ in ] argc - Number of command line arguments
[ in ] argv - Array containing command line arguments
Return Type : DWORD
Returns EXIT_SUCCESS if it is successful,
returns EXIT_FAILURE otherwise.
--*/
{
BOOL bUsage = FALSE ;
BOOL bRmSw = FALSE ;
DWORD dwBootID = 0;
BOOL bBootIdFound = FALSE;
DWORD dwExitCode = ERROR_SUCCESS;
PMY_BOOT_ENTRY mybootEntry;
PLIST_ENTRY listEntry;
PBOOT_ENTRY bootEntry;
BOOL bBaseVideo = FALSE ;
BOOL bNoGui = FALSE ;
BOOL bSos = FALSE ;
BOOL bMaxmem = FALSE ;
PWINDOWS_OS_OPTIONS pWindowsOptions = NULL ;
TCHAR szOsLoadOptions[MAX_RES_STRING+1] = NULL_STRING ;
TCHAR szTemp[MAX_RES_STRING+1] = NULL_STRING ;
DWORD dwExitcode = 0 ;
TCMDPARSER2 cmdOptions[7];
PTCMDPARSER2 pcmdOption;
SecureZeroMemory(cmdOptions, SIZE_OF_ARRAY(cmdOptions)*sizeof(TCMDPARSER2) );
//main option
pcmdOption = &cmdOptions[0];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_RMSW;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bRmSw;
// usage
pcmdOption = &cmdOptions[1];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_USAGE;
pcmdOption->dwFlags = CP2_USAGE;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bUsage;
//id option
pcmdOption = &cmdOptions[2];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_ID;
pcmdOption->dwFlags = CP_VALUE_MANDATORY | CP2_MANDATORY;
pcmdOption->dwType = CP_TYPE_UNUMERIC;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &dwBootID;
//maxmem option
pcmdOption = &cmdOptions[3];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_MAXMEM;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bMaxmem;
//basvideo option
pcmdOption = &cmdOptions[4];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_BASEVIDEO;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bBaseVideo;
//nogui option
pcmdOption = &cmdOptions[5];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_NOGUIBOOT;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bNoGui;
//sos option
pcmdOption = &cmdOptions[6];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_SOS;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bSos;
// Parsing the copy option switches
if ( !DoParseParam2( argc, argv, 0, SIZE_OF_ARRAY(cmdOptions ), cmdOptions, 0 ) )
{
ShowLastErrorEx( stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return (EXIT_FAILURE);
}
//check if usage is specified with more than one option
if( (TRUE == bUsage) && (argc > 3) )
{
ShowMessage(stderr,GetResString(IDS_INVALID_SYNTAX_RMSW));
return ( EXIT_FAILURE );
}
// Displaying query usage if user specified -? with -query option
if( bUsage )
{
displayRmSwUsage_IA64();
return (EXIT_SUCCESS);
}
dwExitcode = InitializeEFI();
if(EXIT_FAILURE == dwExitcode )
{
return EXIT_FAILURE ;
}
//display an error mesage if none of the options are specified.
if((!bSos)&&(!bBaseVideo)&&(!bNoGui)&&(!bMaxmem))
{
ShowMessage(stderr,GetResString(IDS_INVALID_SYNTAX_RMSW));
return EXIT_FAILURE;
}
//Query the boot entries till u get the BootID specified by the user
for (listEntry = BootEntries.Flink;listEntry != &BootEntries;listEntry = listEntry->Flink)
{
//Get the boot entry
mybootEntry = CONTAINING_RECORD( listEntry, MY_BOOT_ENTRY, ListEntry );
if(mybootEntry->myId == dwBootID)
{
bBootIdFound = TRUE;
bootEntry = &mybootEntry->NtBootEntry;
//Check whether the bootEntry is a Windows one or not.
//The OS load options can be added only to a Windows boot entry.
if(!IsBootEntryWindows(bootEntry))
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_OSOPTIONS),dwBootID);
dwExitCode = EXIT_FAILURE;
break;
}
pWindowsOptions = (PWINDOWS_OS_OPTIONS)bootEntry->OsOptions;
if(StringLengthW(pWindowsOptions->OsLoadOptions, 0) > MAX_RES_STRING)
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_STRING_LENGTH1),MAX_RES_STRING);
return EXIT_FAILURE ;
}
// copy the existing OS Loadoptions into a string.
StringCopy(szOsLoadOptions,pWindowsOptions->OsLoadOptions, SIZE_OF_ARRAY(szOsLoadOptions));
//check if the user has entered -basevideo option
if(bBaseVideo)
{
if (FindString(szOsLoadOptions,BASEVIDEO_VALUE, 0) == 0 )
{
ShowMessage(stderr,GetResString(IDS_NO_BV_SWITCH));
dwExitCode = EXIT_FAILURE;
break;
}
else
{
// remove the basevideo switch from the OS Load Options string.
removeSubString(szOsLoadOptions,BASEVIDEO_VALUE);
}
}
if(bSos)
{
if (FindString(szOsLoadOptions,SOS_VALUE, 0) == 0 )
{
ShowMessage(stderr,GetResString(IDS_NO_SOS_SWITCH));
dwExitCode = EXIT_FAILURE;
break;
}
else
{
// remove the /sos switch from the Load Options string.
removeSubString(szOsLoadOptions,SOS_VALUE);
}
}
if(bNoGui)
{
if (FindString(szOsLoadOptions,NOGUI_VALUE,0) == 0 )
{
ShowMessage(stderr,GetResString(IDS_NO_NOGUI_SWITCH));
dwExitCode = EXIT_FAILURE;
break;
}
else
{
// remove the noguiboot switch to the OS Load Options string.
removeSubString(szOsLoadOptions,NOGUI_VALUE);
}
}
if(bMaxmem)
{
if (FindString(szOsLoadOptions,MAXMEM_VALUE1,0) == 0 )
{
ShowMessage(stderr,GetResString(IDS_NO_MAXMEM_SWITCH));
dwExitCode = EXIT_FAILURE;
break;
}
else
{
// add the redirect switch to the OS Load Options string.
//for, a temporary string of form /maxmem=xx so that it
//can be checked in the Os load options,
if ( GetSubString(szOsLoadOptions,MAXMEM_VALUE1,szTemp) == EXIT_FAILURE)
{
return EXIT_FAILURE ;
}
removeSubString(szOsLoadOptions,szTemp);
if(FindString(szOsLoadOptions,MAXMEM_VALUE1,0)!=0)
{
ShowMessage(stderr,GetResString(IDS_NO_MAXMEM) );
return EXIT_FAILURE ;
}
}
}
//display error message if Os Load options is more than 254
// characters.
if(StringLengthW(szOsLoadOptions,0) > MAX_RES_STRING)
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_STRING_LENGTH1),MAX_RES_STRING);
return EXIT_FAILURE ;
}
//Change the OS load options. Pass NULL to friendly name as we are not changing the same
//szRawString is the Os load options specified by the user
dwExitCode = ChangeBootEntry(bootEntry, NULL, szOsLoadOptions);
if(dwExitCode == ERROR_SUCCESS)
{
ShowMessageEx(stdout, 1, TRUE, GetResString(IDS_SUCCESS_CHANGE_OSOPTIONS),dwBootID);
}
else
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_OSOPTIONS),dwBootID);
}
break;
}
}
if(bBootIdFound == FALSE)
{
//Could not find the BootID specified by the user so output the message and return failure
ShowMessage(stderr,GetResString(IDS_INVALID_BOOTID));
dwExitCode = EXIT_FAILURE;
}
Freelist();
return (dwExitCode);
}
DWORD
ProcessDbg1394Switch_IA64( IN DWORD argc,
IN LPCTSTR argv[]
)
/*++
Routine Description :
Which process the dbg1394 switch
Arguments :
[ in ] argc - Number of command line arguments
[ in ] argv - Array containing command line arguments
Return Type : DWORD
Returns EXIT_SUCCESS if it is successful,
returns EXIT_FAILURE otherwise.
--*/
{
BOOL bUsage = FALSE ;
BOOL bDbg1394 = FALSE ;
DWORD dwBootID = 0;
BOOL bBootIdFound = FALSE;
DWORD dwExitCode = ERROR_SUCCESS;
PMY_BOOT_ENTRY mybootEntry;
PLIST_ENTRY listEntry;
PBOOT_ENTRY bootEntry;
PWINDOWS_OS_OPTIONS pWindowsOptions = NULL ;
TCHAR szOsLoadOptions[MAX_RES_STRING+1] = NULL_STRING ;
TCHAR szTemp[MAX_RES_STRING+1] = NULL_STRING ;
TCHAR szTmpBuffer[MAX_RES_STRING+1] = NULL_STRING ;
TCHAR szChannel[MAX_RES_STRING+1] = NULL_STRING ;
TCHAR szDefault[MAX_RES_STRING+1] = NULL_STRING ;
DWORD dwChannel = 0 ;
DWORD dwCode = 0 ;
DWORD dwExitcode = 0 ;
TCMDPARSER2 cmdOptions[5];
PTCMDPARSER2 pcmdOption;
SecureZeroMemory(cmdOptions, SIZE_OF_ARRAY(cmdOptions)*sizeof(TCMDPARSER2) );
//main option
pcmdOption = &cmdOptions[0];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_DBG1394;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bDbg1394;
//id usage
pcmdOption = &cmdOptions[1];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_USAGE;
pcmdOption->dwFlags = CP2_USAGE;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bUsage;
//default option
pcmdOption = &cmdOptions[2];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_ID;
pcmdOption->dwFlags = CP_VALUE_MANDATORY | CP2_MANDATORY;
pcmdOption->dwType = CP_TYPE_UNUMERIC;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &dwBootID;
//id option
pcmdOption = &cmdOptions[3];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_CHANNEL;
pcmdOption->dwFlags = CP_VALUE_MANDATORY;
pcmdOption->dwType = CP_TYPE_UNUMERIC;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &dwChannel;
//default option
pcmdOption = &cmdOptions[4];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_DEFAULT;
pcmdOption->dwFlags = CP2_DEFAULT | CP2_MANDATORY | CP2_VALUE_TRIMINPUT | CP2_VALUE_NONULL;;
pcmdOption->dwType = CP_TYPE_TEXT;
pcmdOption->dwCount = 1;
pcmdOption->pValue = szDefault;
pcmdOption->dwLength= MAX_STRING_LENGTH;
// Parsing the copy option switches
if ( !DoParseParam2( argc, argv, 0, SIZE_OF_ARRAY(cmdOptions ), cmdOptions, 0 ) )
{
ShowLastErrorEx( stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return (EXIT_FAILURE);
}
//check if usage is specified with more than one option
if( (TRUE == bUsage) && (argc > 3) )
{
ShowMessage(stderr,GetResString(IDS_INVALID_SYNTAX_DBG1394));
return ( EXIT_FAILURE );
}
// Displaying query usage if user specified -? with -query option
if( bUsage )
{
displayDbg1394Usage_IA64() ;
return (EXIT_SUCCESS);
}
dwExitcode = InitializeEFI();
if(EXIT_FAILURE == dwExitcode )
{
return EXIT_FAILURE ;
}
if((cmdOptions[2].dwActuals == 0) &&(dwBootID == 0 ))
{
ShowMessage(stderr,GetResString(IDS_ERROR_ID_MISSING));
ShowMessage(stderr,GetResString(IDS_1394_HELP));
return (EXIT_FAILURE);
}
//
//display error message if user enters a value
// other than on or off
//
if( ( StringCompare(szDefault,OFF_STRING,TRUE,0)!=0 ) && (StringCompare(szDefault,ON_STRING,TRUE,0)!=0 ) )
{
ShowMessage(stderr,GetResString(IDS_ERROR_DEFAULT_MISSING));
ShowMessage(stderr,GetResString(IDS_1394_HELP));
return (EXIT_FAILURE);
}
if(( StringCompare(szDefault,OFF_STRING,TRUE,0)==0 ) && (cmdOptions[3].dwActuals != 0) )
{
ShowMessage(stderr,GetResString(IDS_INVALID_SYNTAX_DBG1394));
return (EXIT_FAILURE);
}
if(( StringCompare(szDefault,ON_STRING,TRUE,0)==0 ) && (cmdOptions[3].dwActuals == 0) )
{
ShowMessage(stderr,GetResString(IDS_MISSING_CHANNEL));
return (EXIT_FAILURE);
}
if(( StringCompare(szDefault,ON_STRING,TRUE,0)==0 ) && (cmdOptions[3].dwActuals != 0) &&( (dwChannel < 1) || (dwChannel > 64 ) ) )
{
ShowMessage(stderr,GetResString(IDS_INVALID_CH_RANGE));
return (EXIT_FAILURE);
}
//Query the boot entries till u get the BootID specified by the user
for (listEntry = BootEntries.Flink;listEntry != &BootEntries;listEntry = listEntry->Flink)
{
//Get the boot entry
mybootEntry = CONTAINING_RECORD( listEntry, MY_BOOT_ENTRY, ListEntry );
if(mybootEntry->myId == dwBootID)
{
bBootIdFound = TRUE;
bootEntry = &mybootEntry->NtBootEntry;
//Check whether the bootEntry is a Windows one or not.
//The OS load options can be added only to a Windows boot entry.
if(!IsBootEntryWindows(bootEntry))
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_OSOPTIONS),dwBootID);
dwExitCode = EXIT_FAILURE;
break;
}
pWindowsOptions = (PWINDOWS_OS_OPTIONS)bootEntry->OsOptions;
if(StringLengthW(pWindowsOptions->OsLoadOptions,0) > MAX_RES_STRING)
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_STRING_LENGTH1),MAX_RES_STRING);
return EXIT_FAILURE ;
}
// copy the existing OS Loadoptions into a string.
StringCopy(szOsLoadOptions,pWindowsOptions->OsLoadOptions, SIZE_OF_ARRAY(szOsLoadOptions));
//check if the user has entered on option
if(StringCompare(szDefault,ON_STRING,TRUE,0)==0 )
{
if(FindString(szOsLoadOptions,DEBUGPORT,0) != 0)
{
ShowMessage(stderr,GetResString(IDS_DUPLICATE_ENTRY));
dwExitCode = EXIT_FAILURE;
break;
}
if(FindString(szOsLoadOptions,BAUD_TOKEN,0) != 0)
{
ShowMessage(stderr,GetResString(IDS_ERROR_BAUD_RATE));
dwExitCode = EXIT_FAILURE ;
break;
}
if( FindString(szOsLoadOptions,DEBUG_SWITCH,0)== 0)
{
if( StringLength(szOsLoadOptions,0) != 0 )
{
StringCopy(szTmpBuffer,TOKEN_EMPTYSPACE, SIZE_OF_ARRAY(szTmpBuffer));
StringConcat(szTmpBuffer,DEBUG_SWITCH, SIZE_OF_ARRAY(szOsLoadOptions));
}
else
{
StringCopy(szTmpBuffer,DEBUG_SWITCH, SIZE_OF_ARRAY(szOsLoadOptions));
}
}
if(StringLength(szTmpBuffer,0) == 0)
{
StringCopy(szTmpBuffer,TOKEN_EMPTYSPACE, SIZE_OF_ARRAY(szTmpBuffer));
}
else
{
StringConcat(szTmpBuffer,TOKEN_EMPTYSPACE, SIZE_OF_ARRAY(szTmpBuffer));
}
StringConcat(szTmpBuffer,DEBUGPORT_1394, SIZE_OF_ARRAY(szTmpBuffer)) ;
if(dwChannel!=0)
{
//frame the string and concatenate to the Os Load options.
StringConcat(szTmpBuffer,TOKEN_EMPTYSPACE,SIZE_OF_ARRAY(szTmpBuffer));
StringConcat(szTmpBuffer,TOKEN_CHANNEL,SIZE_OF_ARRAY(szTmpBuffer));
StringConcat(szTmpBuffer,TOKEN_EQUAL,SIZE_OF_ARRAY(szTmpBuffer));
_ltow(dwChannel,szChannel,10);
StringConcat(szTmpBuffer,szChannel,SIZE_OF_ARRAY(szTmpBuffer));
}
}
if(StringCompare(szDefault,OFF_STRING,TRUE,0)==0 )
{
if(FindString(szOsLoadOptions,DEBUGPORT_1394,0) == 0)
{
ShowMessage(stderr,GetResString(IDS_NO_1394_SWITCH));
dwExitCode = EXIT_FAILURE;
break;
}
//
//remove the port from the Os Load options string.
//
removeSubString(szOsLoadOptions,DEBUGPORT_1394);
// check if the string contains the channel token
// and if present remove that also.
//
if(FindString(szOsLoadOptions,TOKEN_CHANNEL,0)!=0)
{
StringCopy(szTemp,NULL_STRING, MAX_RES_STRING);
dwCode = GetSubString(szOsLoadOptions,TOKEN_CHANNEL,szTemp);
if(dwCode == EXIT_SUCCESS)
{
//
//Remove the channel token if present.
//
if(StringLengthW(szTemp,0)!= 0)
{
removeSubString(szOsLoadOptions,szTemp);
removeSubString(szOsLoadOptions,DEBUG_SWITCH);
}
}
}
removeSubString(szOsLoadOptions,DEBUG_SWITCH);
}
//display error message if Os Load options is more than 254
// characters.
if(StringLengthW(szOsLoadOptions,0)+StringLength(szTmpBuffer,0) > MAX_RES_STRING)
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_STRING_LENGTH1),MAX_RES_STRING);
return EXIT_FAILURE ;
}
else
{
StringConcat(szOsLoadOptions, szTmpBuffer, SIZE_OF_ARRAY(szOsLoadOptions) );
}
//Change the OS load options. Pass NULL to friendly name as we are not changing the same
//szRawString is the Os load options specified by the user
dwExitCode = ChangeBootEntry(bootEntry, NULL, szOsLoadOptions);
if(dwExitCode == ERROR_SUCCESS)
{
ShowMessageEx(stdout, 1, TRUE, GetResString(IDS_SUCCESS_CHANGE_OSOPTIONS),dwBootID);
}
else
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_OSOPTIONS),dwBootID);
}
break;
}
}
if(bBootIdFound == FALSE)
{
//Could not find the BootID specified by the user so output the message and return failure
ShowMessage(stderr,GetResString(IDS_INVALID_BOOTID));
dwExitCode = EXIT_FAILURE;
}
Freelist();
return (dwExitCode);
}
DWORD
ProcessMirrorSwitch_IA64( IN DWORD argc,
IN LPCTSTR argv[]
)
/*++
Routine Description :
process the mirror switch
Arguments :
[ in ] argc - Number of command line arguments
[ in ] argv - Array containing command line arguments
Return Type : DWORD
Returns EXIT_SUCCESS if it is successful,
returns EXIT_FAILURE otherwise.
--*/
{
BOOL bUsage = FALSE ;
DWORD dwBootID = 0;
BOOL bBootIdFound = FALSE;
DWORD dwExitCode = ERROR_SUCCESS;
PMY_BOOT_ENTRY mybootEntry;
TCHAR szAdd[MAX_RES_STRING+1] = NULL_STRING ;
TCHAR szLoaderPath[MAX_RES_STRING+1] = NULL_STRING ;
BOOL bMirror = FALSE ;
NTSTATUS status ;
DWORD error = 0 ;
TCHAR szFinalStr[256] = NULL_STRING ;
TCHAR szResult[MAX_RES_STRING+1] = NULL_STRING ;
DWORD dwActuals = 0 ;
PBOOT_ENTRY_LIST ntBootEntries = NULL;
PBOOT_ENTRY pBootEntry = NULL ;
PFILE_PATH pFilePath = NULL ;
PWSTR NtFilePath ;
PTCHAR szPartition = NULL ;
TCHAR szOsLoaderPath[MAX_RES_STRING+1] = NULL_STRING ;
TCHAR szSystemPartition[MAX_RES_STRING+1] = NULL_STRING ;
TCHAR szBrackets[] = _T("{}");
PLIST_ENTRY listEntry;
BOOL bFlag = TRUE ;
TCHAR szFriendlyName[MAX_STRING_LENGTH] = NULL_STRING ;
DWORD dwExitcode = 0 ;
TCMDPARSER2 cmdOptions[5];
PTCMDPARSER2 pcmdOption;
BOOL bNobreak = TRUE;
SecureZeroMemory(cmdOptions, SIZE_OF_ARRAY(cmdOptions)*sizeof(TCMDPARSER2) );
//main option
pcmdOption = &cmdOptions[0];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_MIRROR;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bMirror;
//id usage
pcmdOption = &cmdOptions[1];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_USAGE;
pcmdOption->dwFlags = CP2_USAGE;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bUsage;
// add option
pcmdOption = &cmdOptions[2];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_ADD;
pcmdOption->dwFlags = CP_VALUE_MANDATORY | CP2_VALUE_TRIMINPUT | CP2_VALUE_NONULL | CP2_MANDATORY;
pcmdOption->dwType = CP_TYPE_TEXT;
pcmdOption->dwCount = 1;
pcmdOption->pValue = szAdd;
pcmdOption->dwLength= MAX_STRING_LENGTH;
//id option
pcmdOption = &cmdOptions[3];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_ID;
pcmdOption->dwFlags = CP_VALUE_MANDATORY;
pcmdOption->dwType = CP_TYPE_UNUMERIC;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &dwBootID;
// friendly option
pcmdOption = &cmdOptions[4];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = SWITCH_DESCRIPTION;
pcmdOption->dwFlags = CP_VALUE_MANDATORY | CP2_VALUE_TRIMINPUT;
pcmdOption->dwType = CP_TYPE_TEXT;
pcmdOption->dwCount = 1;
pcmdOption->pValue = szFriendlyName;
pcmdOption->dwLength= MAX_STRING_LENGTH;
// Parsing the copy option switches
if ( !(DoParseParam2( argc, argv, 0, SIZE_OF_ARRAY(cmdOptions ), cmdOptions, 0 ) ) )
{
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return (EXIT_FAILURE);
}
//check if usage is specified with more than one option
if( (TRUE == bUsage) && (argc > 3) )
{
ShowMessage(stderr,GetResString(IDS_MIRROR_SYNTAX));
return ( EXIT_FAILURE );
}
// Displaying query usage if user specified -? with -query option
if( bUsage )
{
displayMirrorUsage_IA64() ;
return (EXIT_SUCCESS);
}
dwExitcode = InitializeEFI();
if(EXIT_FAILURE == dwExitcode )
{
return EXIT_FAILURE ;
}
// If the user enters empty string after add option then display an error
// message.
TrimString(szAdd,TRIM_ALL);
TrimString(szFriendlyName,TRIM_ALL);
//
//copy the default friendly name from resource file if no
//friendly name is specified.
//
if(cmdOptions[4].dwActuals == 0)
{
StringCopy(szFriendlyName,GetResString(IDS_MIRROR_NAME), SIZE_OF_ARRAY(szFriendlyName));
}
if(StringLengthW(szAdd,0) !=0)
{
//Trim of the Brackets which may be specified
//along with the GUID.
TrimString2(szAdd, szBrackets, TRIM_ALL);
dwActuals = 0 ;
//get the ARC signature path corresponding to the GUID specified.
if (GetDeviceInfo(szAdd,szFinalStr,0,dwActuals) == EXIT_FAILURE )
{
return EXIT_FAILURE ;
}
StringConcat(szFinalStr,_T("\\WINDOWS"), SIZE_OF_ARRAY(szFinalStr));
//
//if the user does specifies /id option
// then retreive the OS Load Path from the
// registry
//
if(cmdOptions[3].dwActuals == 0 )
{
//retreive the Os Loader Path from the registry.
if(GetBootPath(IDENTIFIER_VALUE2,szResult) != ERROR_SUCCESS )
{
ShowMessage(stderr,GetResString(IDS_ERROR_UNEXPECTED));
return EXIT_FAILURE ;
}
//retreive the Os Loader Path from the registry.
if( GetBootPath(IDENTIFIER_VALUE3,szLoaderPath)!= ERROR_SUCCESS )
{
ShowMessage(stderr,GetResString(IDS_ERROR_UNEXPECTED));
return EXIT_FAILURE ;
}
bFlag = TRUE ;
//call the function which adds the mirror plex.
if (AddMirrorPlex(szFinalStr,szLoaderPath,szResult,bFlag,szFriendlyName) == EXIT_FAILURE )
{
return EXIT_FAILURE ;
}
}
else
{
// query the information from the NVRAM .
status = BootCfg_EnumerateBootEntries(&ntBootEntries);
if( !NT_SUCCESS(status) )
{
error = RtlNtStatusToDosError( status );
return EXIT_FAILURE ;
}
for (listEntry = BootEntries.Flink; listEntry != &BootEntries; listEntry = listEntry->Flink)
{
//Get the boot entry
mybootEntry = CONTAINING_RECORD( listEntry, MY_BOOT_ENTRY, ListEntry );
//check for the id specified by the user matches the
// id
if(mybootEntry->myId == dwBootID)
{
bBootIdFound = TRUE;
pBootEntry = &mybootEntry->NtBootEntry;
//Check whether the bootEntry is a Windows one or not.
//The OS load options can be added only to a Windows boot entry.
if(!IsBootEntryWindows(pBootEntry))
{
ShowMessageEx(stderr, 1, TRUE, GetResString(IDS_ERROR_OSOPTIONS),dwBootID);
dwExitCode = EXIT_FAILURE;
break;
}
}
}
// display an error
if (pBootEntry == NULL)
{
ShowMessage(stderr,GetResString(IDS_PARTITION_ERROR));
SAFEFREE(ntBootEntries);
return EXIT_FAILURE ;
}
//Get a pointer to the FILE_PATH structure.
pFilePath = (PFILE_PATH)ADD_OFFSET(pBootEntry, BootFilePathOffset);
//get the name of the .
NtFilePath = GetNtNameForFilePath(pFilePath );
if(NtFilePath == NULL)
{
ShowMessage(stderr,GetResString(IDS_ERROR_MIRROR));
SAFEFREE(ntBootEntries);
return EXIT_FAILURE ;
}
// split the path to get the SystemPartition path and the
// OsLoader Path .
szPartition = _tcstok(NtFilePath,_T("\\"));
//display error message and exit if szPartition is null.
if(szPartition == NULL)
{
ShowMessage(stderr,GetResString(IDS_TOKEN_ABSENT));
SAFEFREE(ntBootEntries);
return EXIT_FAILURE ;
}
//concatenate the "\" to frame the path .
StringConcat(szOsLoaderPath,_T("\\"), SIZE_OF_ARRAY(szOsLoaderPath));
StringConcat(szOsLoaderPath,szPartition, SIZE_OF_ARRAY(szOsLoaderPath) );
StringConcat(szOsLoaderPath,_T("\\"), SIZE_OF_ARRAY(szOsLoaderPath));
szPartition = _tcstok(NULL,_T("\\"));
//display error message and exit if szPartition is null.
if(NULL == szPartition )
{
ShowMessage(stderr,GetResString(IDS_TOKEN_ABSENT));
SAFEFREE(ntBootEntries);
return EXIT_FAILURE ;
}
StringConcat(szOsLoaderPath,szPartition, SIZE_OF_ARRAY(szOsLoaderPath));
//Framing the OsLoader Path
do
{
szPartition = _tcstok(NULL,_T("\\"));
if(szPartition == NULL)
{
break ;
bNobreak = FALSE;
}
StringConcat(szSystemPartition,_T("\\"), SIZE_OF_ARRAY(szSystemPartition));
StringConcat(szSystemPartition,szPartition, SIZE_OF_ARRAY(szSystemPartition));
}while(TRUE == bNobreak );
//This flag is for determining if the boot path should be BOOTFILE_PATH1
//or BOOTFILE_PATH
bFlag = FALSE ;
//call the function which adds the mirror plex.
if ( AddMirrorPlex(szFinalStr,szSystemPartition,szOsLoaderPath,bFlag,szFriendlyName) == EXIT_FAILURE )
{
return EXIT_FAILURE ;
}
}
}
SAFEFREE(ntBootEntries);
Freelist();
return EXIT_SUCCESS ;
}
NTSTATUS
FindBootEntry(IN PVOID pEntryListHead,
IN WCHAR *pwszTarget,
OUT PBOOT_ENTRY *ppTargetEntry
)
/*++
Routine description : Routine finds a boot entry in the list of all boot
entries and returns a pointer into the list for the found entry.
Arguments:
pEntryListHead - The address of a pointer to a BOOT_ENTRY_LIST struct.
pwszTarget - The OsLoadPath (install path) string.
"signature(<part GUID>-<part#>-<part_start>-<part_len>)"
OR
"signature(<part GUID>-<part#>-<part_start>-<part_len>)\\WINDOWS"
on input. If we find the entry in NVRAM, we copy the
full install path back to the input string so that it includes
the directory name.
ppTargetEntry - The address of a BOOT_ENTRY pointer, points to the
found entry at return.
Return Value : NT status
STATUS_INSUFFICIENT_RESOURCES
STATUS_ACCESS_VIOLATION
STATUS_UNSUPPORTED
STATUS_OBJECT_NAME_NOT_FOUND
STATUS_SUCCESS and *ppTargetEntry should be non-NULL for success.
--*/
{
LONG status = STATUS_SUCCESS;
PBOOT_ENTRY_LIST pEntryList = NULL;
PBOOT_ENTRY pEntry = NULL;
PWINDOWS_OS_OPTIONS pOsOptions = NULL;
PFILE_PATH pTransEntry = NULL;
DWORD dwTransSize = 0L;
DWORD i = 0L;
BOOL bFlag = FALSE ;
BOOL bNobreak = FALSE ;
DWORD dwCount = 0L ;
TCHAR szFinalStr[256] = NULL_STRING ;
DWORD dwFailCount = 0L;
DWORD dwSuccessCount = 0L;
if ( !pEntryListHead || !pwszTarget || !ppTargetEntry )
{
ShowMessage(stderr,GetResString(IDS_FIND_BOOT_ENTRY) );
return STATUS_INVALID_PARAMETER;
}
*ppTargetEntry = NULL;
pEntryList = (PBOOT_ENTRY_LIST) pEntryListHead;
//
// Iterate over all the entries returned looking for the target
// boot partition's entry. Convert the install path for each
// entry to the signature format, then compare to the
// input partition signature formatted path.
//
bNobreak = TRUE;
do
{
//
// Translate the entry's install path to signature format.
//
if ( pEntryList )
{
pEntry = &pEntryList->BootEntry;
}
else
{
ShowMessage(stderr,GetResString(IDS_FIND_BOOT_ENTRY_NULL));
status = STATUS_OBJECT_NAME_NOT_FOUND;
bNobreak = FALSE;
break;
}
//
// If this entry does not have the BOOT_ENTRY_ATTRIBUTE_WINDOWS
// attribute set, or, the attribute is set and this entry has
// an invalid OsOptions structure length, move to the next entry
// and continue searching and check the next boot entry
//
if ( !(pEntry->Attributes & BOOT_ENTRY_ATTRIBUTE_WINDOWS) || ( (pEntry->Attributes & BOOT_ENTRY_ATTRIBUTE_WINDOWS) && pEntry->OsOptionsLength < sizeof(WINDOWS_OS_OPTIONS) ) )
{
//exit from the loop if we have reached the last Boot Entry.
if ( !pEntryList->NextEntryOffset )
{
bNobreak = FALSE;
break;
}
//
// Continue with the next iteration
// after obtaining the pointer to the next entry.
//
pEntryList = (PBOOT_ENTRY_LIST)(((PBYTE)pEntryList) + pEntryList->NextEntryOffset);
continue;
}
//
// Use the entry's current length to start and resize
// if necessary.
//
dwTransSize = pEntry->Length;
for ( i = 1; i <= 2; i++ )
{
if ( pTransEntry )
{
MEMFREE(pTransEntry);
pTransEntry = NULL;
}
pTransEntry = (PFILE_PATH) MEMALLOC(dwTransSize);
if ( !pTransEntry )
{
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_SYSTEM );
status = STATUS_NO_MEMORY;
SAFEMEMFREE(pTransEntry);
return status ;
}
pOsOptions = (WINDOWS_OS_OPTIONS *)&pEntry->OsOptions;
status = NtTranslateFilePath
(
(PFILE_PATH)( ((PBYTE) pOsOptions) + pOsOptions->OsLoadPathOffset ),
FILE_PATH_TYPE_ARC_SIGNATURE,
pTransEntry,
&dwTransSize
);
if ( STATUS_INSUFFICIENT_RESOURCES == status )
{
continue;
}
else
{
break;
}
}
//
// Ignore STATUS_OBJECT_NAME_NOT_FOUND
// We shouldn't get that error anyway, since we are using
// the long signature format.
//
if ( !NT_SUCCESS(status)&& STATUS_OBJECT_NAME_NOT_FOUND != status )
{
DISPLAY_MESSAGE( stderr,GetResString(IDS_TRANSLATE_FAIL));
SAFEMEMFREE(pTransEntry);
return status ;
}
//
// Compare this entry's install path to the current boot
// partition's signature formatted install path.
// If the input install path may not include the install
// directory name.
//
//
// Check if the GUID specified by the User matches with the set of GUID's
// already present
//
if ( NT_SUCCESS(status) && !(wcsncmp( (WCHAR*)&pTransEntry->FilePath, pwszTarget, 48 ) ) )
{
// Set the flag to true indicating that the
// specified GUID matches
//
bFlag = TRUE ;
//
// Check if the ARC Path specified by the User matches with the ARC Path
// of the entry already present and if so display an error message and exit.
//
if( !(StringCompare( (WCHAR*)&pTransEntry->FilePath, pwszTarget, TRUE, StringLengthW(pwszTarget,0) ) ) )
{
ShowMessage(stderr,GetResString(IDS_ALREADY_UPDATED));
SAFEMEMFREE(pTransEntry) ;
return STATUS_OBJECT_NAME_NOT_FOUND;
}
else
{
*ppTargetEntry = pEntry;
//concatenate the string "\WINDOWS" to the path formed.
StringCopy ( szFinalStr, NULL_STRING, SIZE_OF_ARRAY(szFinalStr));
StringConcat(szFinalStr,pwszTarget,SIZE_OF_ARRAY(szFinalStr));
StringConcat(szFinalStr,_T("\\WINDOWS"), SIZE_OF_ARRAY(szFinalStr));
//
//modify the Boot Entry with the Arc Signature path specified.
//
status = ModifyBootEntry(szFinalStr,*ppTargetEntry);
if ( !NT_SUCCESS(status) )
{
//If unsuccessful to update the Boot_Entry then increment the Failure
//count.
dwFailCount++ ;
}
else
{
//If successfully updated the Boot_Entry then increment the Success
//count.
dwSuccessCount++;
}
if ( !pEntryList->NextEntryOffset )
{
bNobreak = FALSE;
break;
}
else
{
pEntryList = (PBOOT_ENTRY_LIST)( ((PBYTE)pEntryList) + pEntryList->NextEntryOffset );
continue ;
}
}
}
else
{
if ( !pEntryList->NextEntryOffset )
{
bNobreak = FALSE;
break;
}
pEntryList = (PBOOT_ENTRY_LIST)( ((PBYTE)pEntryList) + pEntryList->NextEntryOffset );
}
}while(TRUE == bNobreak );
// Depending upon the number of entries successfully updated
// display appropriate messages.
if((0 != dwFailCount)&&(0 == dwSuccessCount))
{
ShowMessage(stdout,GetResString(IDS_MODIFY_FAIL));
}
if(( 0 != dwSuccessCount )&&(0 == dwFailCount))
{
ShowMessage(stdout,GetResString(IDS_GUID_MODIFIED));
}
else if( ( 0 != dwSuccessCount )&&(0 != dwFailCount))
{
ShowMessage(stdout,GetResString(IDS_PARTIAL_UPDATE));
}
//display an error message if the GUID specified does not match with the GUID's present.
if(FALSE == bFlag )
{
//ShowMessage(stderr,GetResString(IDS_FIND_FAIL));
SAFEMEMFREE(pTransEntry) ;
return STATUS_INVALID_PARAMETER;
}
SAFEMEMFREE(pTransEntry)
return status;
}
LPVOID
MEMALLOC( ULONG size )
/*++
Routine Description : Allocates the memory Block.
Arguments :
[ in ] block : Size of the block to be allocated.
Return Type : LPVOID
--*/
{
HANDLE hProcessHeap;
hProcessHeap = GetProcessHeap();
if (hProcessHeap == NULL ||
size > 0x7FFF8) {
return NULL;
}
else {
return HeapAlloc (hProcessHeap, HEAP_ZERO_MEMORY, size);
}
}
VOID MEMFREE ( LPVOID block ) {
/*++
Routine Description : Frees the memory Allocated.
Arguments :
[ in ] block : Block to be freed.
Return Type : VOID
--*/
HANDLE hProcessHeap;
hProcessHeap = GetProcessHeap();
if (hProcessHeap != NULL) {
HeapFree(hProcessHeap, 0, block);
}
}
NTSTATUS
ModifyBootEntry( IN WCHAR *pwszInstallPath,
IN PBOOT_ENTRY pSourceEntry
)
/*++
Routine description : This routine is used to modify a boot entry in the NVRAM.
Arguments:
pwszInstallPath - The new install path
pSourceEntry - Entry that we will modify
Return Value : NT status
--*/
{
LONG status = STATUS_SUCCESS;
PFILE_PATH pLoaderFile = NULL;
ULONG dwLoaderFileSize = 0L;
PFILE_PATH pInstallPath = NULL; // new install path
ULONG dwInstallPathSize = 0L; // new install path size
PWINDOWS_OS_OPTIONS pWinOpt = NULL;
ULONG dwWinOptSize = 0L;
PBOOT_ENTRY pSetEntry = 0L; // new, modified entry
ULONG dwSetEntrySize = 0L;
DWORD dwFriendlyNameSize = 0L;
DWORD dwAlign = 0L;
PWINDOWS_OS_OPTIONS pSourceWinOpt = NULL; // old, source entry options to be modified
PFILE_PATH pSourceInstallPath = NULL; // old, source entry install path to be modified
//
// Validate params.
//
if ( !pwszInstallPath
|| !(StringLengthW(pwszInstallPath,0))
|| !pSourceEntry )
{
return STATUS_INVALID_PARAMETER;
}
//
// Setup the BootFilePath member of the BOOT_ENTRY.
//
dwLoaderFileSize = ( (PFILE_PATH) (((PBYTE)pSourceEntry) + pSourceEntry->BootFilePathOffset) )->Length;
pLoaderFile = (PFILE_PATH)MEMALLOC(dwLoaderFileSize);
if ( NULL == pLoaderFile )
{
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return STATUS_NO_MEMORY ;
}
RtlCopyMemory(pLoaderFile,((PBYTE)pSourceEntry) + pSourceEntry->BootFilePathOffset,
dwLoaderFileSize
);
//
// Setup the OsLoadPath member of the WINDOWS_OS_OPTIONS struct.
//
dwInstallPathSize = FIELD_OFFSET(FILE_PATH, FilePath) + ( (StringLengthW(pwszInstallPath,0)+1) * sizeof(WCHAR) );
pInstallPath = (PFILE_PATH)MEMALLOC(dwInstallPathSize);
if ( NULL == pInstallPath )
{
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
SAFEMEMFREE(pLoaderFile);
return STATUS_NO_MEMORY ;
}
pSourceWinOpt = (PWINDOWS_OS_OPTIONS) &pSourceEntry->OsOptions;
pSourceInstallPath = (PFILE_PATH)( ((PBYTE)pSourceWinOpt)+ pSourceWinOpt->OsLoadPathOffset );
pInstallPath->Version = pSourceInstallPath->Version;
pInstallPath->Length = dwInstallPathSize; // new install path size
pInstallPath->Type = FILE_PATH_TYPE_ARC_SIGNATURE;
RtlCopyMemory(pInstallPath->FilePath, // new path to the OS on the boot partition, "signature(partition_guid)\WINDOWS"
pwszInstallPath,
(StringLengthW(pwszInstallPath,0) + 1) * sizeof(WCHAR)
);
//
// Setup the OsOptions member of the BOOT_ENTRY
//
dwWinOptSize = FIELD_OFFSET(WINDOWS_OS_OPTIONS, OsLoadOptions)
+ ( (StringLengthW(pSourceWinOpt->OsLoadOptions,0) + 1) * sizeof(WCHAR) ) // old OsLoadOptions
+ dwInstallPathSize // new OsLoadPath
+ sizeof(DWORD); // Need to align the FILE_PATH struct
pWinOpt = (PWINDOWS_OS_OPTIONS) MEMALLOC(dwWinOptSize);
if ( NULL == pWinOpt )
{
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
SAFEMEMFREE(pLoaderFile);
SAFEMEMFREE(pInstallPath);
return STATUS_NO_MEMORY ;
}
RtlCopyMemory( pWinOpt->Signature, pSourceWinOpt->Signature, sizeof(WINDOWS_OS_OPTIONS_SIGNATURE) );
pWinOpt->Version = pSourceWinOpt->Version;
pWinOpt->Length = dwWinOptSize;
pWinOpt->OsLoadPathOffset = FIELD_OFFSET(WINDOWS_OS_OPTIONS, OsLoadOptions)
+ ((StringLengthW(pSourceWinOpt->OsLoadOptions,0) + 1) * sizeof(WCHAR));
//
// Need to align the OsLoadPathOffset on a 4 byte boundary.
//
dwAlign = ( pWinOpt->OsLoadPathOffset & (sizeof(DWORD) - 1) );
if ( dwAlign != 0 )
{
pWinOpt->OsLoadPathOffset += sizeof(DWORD) - dwAlign;
}
StringCopy(pWinOpt->OsLoadOptions, pSourceWinOpt->OsLoadOptions, (StringLengthW(pSourceWinOpt->OsLoadOptions,0)));
RtlCopyMemory( ((PBYTE)pWinOpt) + pWinOpt->OsLoadPathOffset, pInstallPath, dwInstallPathSize );
//
// Setup the BOOT_ENTRY struct.
//
dwFriendlyNameSize = ( StringLengthW( (WCHAR *)(((PBYTE)pSourceEntry) + pSourceEntry->FriendlyNameOffset), 0 ) + 1)*sizeof(WCHAR);
dwSetEntrySize = FIELD_OFFSET(BOOT_ENTRY, OsOptions)
+ dwWinOptSize // OsOptions
+ dwFriendlyNameSize // FriendlyName including the NULL terminator
+ dwLoaderFileSize // BootFilePath
+ sizeof(WCHAR) // Need to align the FriendlyName on WCHAR
+ sizeof(DWORD); // Need to align the BootFilePath on DWORD
pSetEntry = (PBOOT_ENTRY) MEMALLOC(dwSetEntrySize);
if ( NULL == pSetEntry )
{
SAFEMEMFREE(pLoaderFile);
SAFEMEMFREE(pInstallPath);
SAFEMEMFREE(pWinOpt);
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return STATUS_NO_MEMORY;
}
pSetEntry->Version = pSourceEntry->Version;
pSetEntry->Length = dwSetEntrySize;
pSetEntry->Id = pSourceEntry->Id; // not used, output param
pSetEntry->Attributes = pSourceEntry->Attributes;
pSetEntry->FriendlyNameOffset = FIELD_OFFSET(BOOT_ENTRY, OsOptions)
+ dwWinOptSize;
//
// Need to align the unicode string on a 2 byte boundary.
//
dwAlign = ( pSetEntry->FriendlyNameOffset & (sizeof(WCHAR) - 1) );
if ( dwAlign != 0 )
{
pSetEntry->FriendlyNameOffset += sizeof(WCHAR) - dwAlign;
}
pSetEntry->BootFilePathOffset = pSetEntry->FriendlyNameOffset + dwFriendlyNameSize;
//
// Need to align the FILE_PATH struct on a 4 byte boundary.
//
dwAlign = ( pSetEntry->BootFilePathOffset & (sizeof(DWORD) - 1) );
if ( dwAlign != 0 )
{
pSetEntry->BootFilePathOffset += sizeof(DWORD) - dwAlign;
}
pSetEntry->OsOptionsLength = dwWinOptSize;
RtlCopyMemory( pSetEntry->OsOptions, pWinOpt, dwWinOptSize );
RtlCopyMemory( ((PBYTE)pSetEntry) + pSetEntry->FriendlyNameOffset,
((PBYTE)pSourceEntry) + pSourceEntry->FriendlyNameOffset,
dwFriendlyNameSize
);
RtlCopyMemory( ((PBYTE)pSetEntry) + pSetEntry->BootFilePathOffset,
pLoaderFile,
dwLoaderFileSize
);
status = NtModifyBootEntry( pSetEntry );
if(!NT_SUCCESS(status))
{
ShowMessage(stderr,GetResString(IDS_MODIFY_FAIL));
}
SAFEMEMFREE(pLoaderFile);
SAFEMEMFREE(pInstallPath);
SAFEMEMFREE(pWinOpt);
SAFEMEMFREE(pSetEntry);
return status;
}
DWORD
ListDeviceInfo(DWORD dwDriveNum)
/*++
Routine description : This routine is used to retrieve the list of device partitions.
Arguments:
szGUID : The address of a pointer to a BOOT_ENTRY_LIST struct.
szFinalStr : The String containing the final ARG signature path.
Return Value : DWORD
EXIT_SUCCESS if it is successful,
EXIT_FAILURE otherwise.
--*/
{
HRESULT hr = S_OK;
HANDLE hDevice ;
BOOL bResult = FALSE ;
PPARTITION_INFORMATION_EX pInfo=NULL ;
PDRIVE_LAYOUT_INFORMATION_EX Drive=NULL;
DWORD dwBytesCount = 0 ;
TCHAR szDriveName[MAX_RES_STRING+1] = NULL_STRING ;
DWORD dwStructSize = 0 ;
TCHAR szInstallPath[MAX_RES_STRING+1] = NULL_STRING;
TCHAR szWindowsDirectory[MAX_PATH*2] = NULL_STRING;
PTCHAR pszTok = NULL ;
PPARTITION_INFORMATION_GPT pGptPartition=NULL;
DWORD dwPartitionId = 0 ;
CHAR szTempBuffer[ 33 ] = "\0";
WCHAR wszOffsetStr[ 33 ] = L"\0";
CHAR szTempBuffer1[ 33 ] = "\0";
WCHAR wszPartitionStr[ 33 ] = L"\0";
WCHAR szOutputStr[MAX_RES_STRING+1] = NULL_STRING ;
NTSTATUS ntstatus;
SecureZeroMemory(szDriveName, SIZE_OF_ARRAY(szDriveName));
hr = StringCchPrintf(szDriveName, SIZE_OF_ARRAY(szDriveName), _T("\\\\.\\physicaldrive%d"),dwDriveNum);
//get a handle after opening the File.
hDevice = CreateFile(szDriveName,
GENERIC_READ|GENERIC_WRITE,
FILE_SHARE_READ|FILE_SHARE_WRITE,
NULL,
OPEN_EXISTING,
0,
NULL);
if(hDevice == INVALID_HANDLE_VALUE)
{
ShowMessage(stderr,GetResString(IDS_INVALID_DISK));
return EXIT_FAILURE ;
}
Drive = (PDRIVE_LAYOUT_INFORMATION_EX)AllocateMemory(sizeof(DRIVE_LAYOUT_INFORMATION_EX) +5000) ;
if(NULL == Drive)
{
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
CloseHandle(hDevice);
return EXIT_FAILURE ;
}
dwStructSize = sizeof(DRIVE_LAYOUT_INFORMATION_EX) ;
bResult = DeviceIoControl(
hDevice,
IOCTL_DISK_GET_DRIVE_LAYOUT_EX,
NULL,
0,
Drive,
sizeof(DRIVE_LAYOUT_INFORMATION_EX)+5000,
&dwBytesCount,
NULL);
if(bResult ==0)
{
SAFEFREE(Drive);
DISPLAY_MESSAGE( stderr, ERROR_TAG);
ShowLastError(stderr);
CloseHandle(hDevice);
return EXIT_FAILURE ;
}
ShowMessageEx(stdout, 1, TRUE, GetResString(IDS_LIST0),dwDriveNum);
ShowMessage(stdout,GetResString(IDS_LIST1));
for(dwPartitionId = 0 ;dwPartitionId < Drive->PartitionCount ; dwPartitionId++)
{
//get a pointer to the corresponding partition.
pInfo = (PPARTITION_INFORMATION_EX)(&Drive->PartitionEntry[dwPartitionId] ) ;
ShowMessageEx(stdout,1, TRUE, GetResString(IDS_LIST2),dwPartitionId+1);
switch(Drive->PartitionStyle )
{
case PARTITION_STYLE_MBR :
ShowMessage(stdout,GetResString(IDS_LIST3));
break;
case PARTITION_STYLE_GPT :
ShowMessage(stdout,GetResString(IDS_LIST4));
break;
case PARTITION_STYLE_RAW :
ShowMessage(stdout,GetResString(IDS_LIST5));
break;
}
ntstatus = RtlLargeIntegerToChar( &pInfo->StartingOffset, 10, SIZE_OF_ARRAY( szTempBuffer ), szTempBuffer );
if ( ! NT_SUCCESS( ntstatus ) )
{
SAFEFREE(Drive);
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_SYSTEM );
CloseHandle(hDevice);
return EXIT_FAILURE ;
}
ntstatus = RtlLargeIntegerToChar( &pInfo->PartitionLength, 10, SIZE_OF_ARRAY( szTempBuffer1 ), szTempBuffer1 );
if ( ! NT_SUCCESS( ntstatus ) )
{
SAFEFREE(Drive);
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_SYSTEM );
CloseHandle(hDevice);
return EXIT_FAILURE ;
}
MultiByteToWideChar( _DEFAULT_CODEPAGE, 0, szTempBuffer, -1, wszOffsetStr, SIZE_OF_ARRAY(wszOffsetStr) );
if( ConvertintoLocale( wszOffsetStr,szOutputStr )== EXIT_FAILURE )
{
SAFEFREE(Drive);
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_SYSTEM );
CloseHandle(hDevice);
return EXIT_FAILURE ;
}
ShowMessage(stdout,GetResString(IDS_LIST6));
ShowMessage(stdout,_X3(szOutputStr));
ShowMessage(stdout,L"\n");
MultiByteToWideChar( _DEFAULT_CODEPAGE, 0, szTempBuffer1, -1, wszPartitionStr, SIZE_OF_ARRAY(wszOffsetStr) );
if( ConvertintoLocale( wszPartitionStr,szOutputStr )== EXIT_FAILURE )
{
SAFEFREE(Drive);
DISPLAY_MESSAGE( stderr, ERROR_TAG);
ShowLastError(stderr);
CloseHandle(hDevice);
return EXIT_FAILURE ;
}
ShowMessage(stdout,GetResString(IDS_LIST7));
ShowMessage(stdout,_X3(szOutputStr));
ShowMessage(stdout,L"\n");
//get a pointer to the PARTITION_INFORMATION_GPT structure.
pGptPartition = AllocateMemory( sizeof( PARTITION_INFORMATION_GPT));
if(NULL == pGptPartition )
{
SAFEFREE(Drive);
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
CloseHandle(hDevice);
return EXIT_FAILURE ;
}
CopyMemory(pGptPartition,&pInfo->Gpt,sizeof(PARTITION_INFORMATION_GPT) );
ShowMessage(stdout, GetResString(IDS_LIST8));
ShowMessageEx(stdout, 11, TRUE, _T("{%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x}"),
pGptPartition->PartitionId.Data1,
pGptPartition->PartitionId.Data2,
pGptPartition->PartitionId.Data3,
pGptPartition->PartitionId.Data4[0],
pGptPartition->PartitionId.Data4[1],
pGptPartition->PartitionId.Data4[2],
pGptPartition->PartitionId.Data4[3],
pGptPartition->PartitionId.Data4[4],
pGptPartition->PartitionId.Data4[5],
pGptPartition->PartitionId.Data4[6],
pGptPartition->PartitionId.Data4[7] );
ShowMessage(stdout, GetResString(IDS_LIST9));
ShowMessageEx(stdout, 11, TRUE,
_T("{%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x}"),
pGptPartition->PartitionType.Data1,
pGptPartition->PartitionType.Data2,
pGptPartition->PartitionType.Data3,
pGptPartition->PartitionType.Data4[0],
pGptPartition->PartitionType.Data4[1],
pGptPartition->PartitionType.Data4[2],
pGptPartition->PartitionType.Data4[3],
pGptPartition->PartitionType.Data4[4],
pGptPartition->PartitionType.Data4[5],
pGptPartition->PartitionType.Data4[6],
pGptPartition->PartitionType.Data4[7] );
//partition name.
if(StringLengthW(pGptPartition->Name,0) != 0)
{
ShowMessageEx(stdout, 1, TRUE, GetResString(IDS_LIST10),pGptPartition->Name);
}
else
{
ShowMessageEx(stdout, 1, TRUE, GetResString(IDS_LIST10),_T("N/A"));
}
}
if( 0 == GetWindowsDirectory(szWindowsDirectory,MAX_PATH) )
{
SAFEFREE(Drive);
SAFEFREE(pGptPartition);
ShowMessage(stderr,GetResString(IDS_ERROR_DRIVE));
CloseHandle(hDevice);
return EXIT_FAILURE ;
}
StringConcat(szWindowsDirectory,_T("*"), SIZE_OF_ARRAY(szWindowsDirectory));
pszTok = _tcstok(szWindowsDirectory,_T("\\"));
if(pszTok == NULL)
{
SAFEFREE(Drive);
SAFEFREE(pGptPartition);
DISPLAY_MESSAGE(stderr,GetResString(IDS_TOKEN_ABSENT));
CloseHandle(hDevice);
return EXIT_FAILURE ;
}
pszTok = _tcstok(NULL,_T("*"));
if(pszTok == NULL)
{
SAFEFREE(Drive);
SAFEFREE(pGptPartition);
DISPLAY_MESSAGE(stderr,GetResString(IDS_TOKEN_ABSENT));
CloseHandle(hDevice);
return EXIT_FAILURE ;
}
SecureZeroMemory(szInstallPath, SIZE_OF_ARRAY(szInstallPath));
hr = StringCchPrintf( szInstallPath, SIZE_OF_ARRAY(szInstallPath),
_T("signature({%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x}-%08x-%016I64x-%016I64x)"),
pGptPartition->PartitionId.Data1,
pGptPartition->PartitionId.Data2,
pGptPartition->PartitionId.Data3,
pGptPartition->PartitionId.Data4[0],
pGptPartition->PartitionId.Data4[1],
pGptPartition->PartitionId.Data4[2],
pGptPartition->PartitionId.Data4[3],
pGptPartition->PartitionId.Data4[4],
pGptPartition->PartitionId.Data4[5],
pGptPartition->PartitionId.Data4[6],
pGptPartition->PartitionId.Data4[7],
pInfo->PartitionNumber,
pInfo->StartingOffset,
pInfo->PartitionLength
);
SAFEFREE(Drive);
SAFEFREE(pGptPartition);
CloseHandle(hDevice);
return EXIT_SUCCESS ;
}
NTSTATUS
AcquirePrivilege( IN CONST ULONG ulPrivilege,
IN CONST BOOLEAN bEnable
)
/*++
Routine description : This routine is used to set or reset a privilege
on a process token.
Arguments:
ulPrivilege - The privilege t enable or disable.
bEnable - TRUE to enable the priviliege, FALSE to disable.
Return Value : NTSTATUS
--*/
{
NTSTATUS status;
BOOLEAN bPrevState;
if ( bEnable ) {
status = RtlAdjustPrivilege( ulPrivilege,
TRUE, // enable
FALSE, // adjust the process token
&bPrevState
);
}
else {
status = RtlAdjustPrivilege( ulPrivilege,
FALSE, // disable
FALSE, // adjust the process token
&bPrevState
);
}
return status;
}
NTSTATUS
EnumerateBootEntries( IN PVOID *ppEntryListHead)
/*++
Routine description : This routine is used to retrieve the list of boot entries.
Arguments:
ppEntryListHead - The address of a pointer to a BOOT_ENTRY_LIST struct.
Return Value : NTSTATUS
--*/
{
LONG status = STATUS_SUCCESS;
DWORD dwEntryListSize = 0x0001000; // 4k
BOOL bNobreak = TRUE;
if ( !ppEntryListHead )
{
ShowMessage(stderr,GetResString(IDS_ERROR_ENUMERATE));
return STATUS_INVALID_PARAMETER;
}
do
{
*ppEntryListHead = (PBOOT_ENTRY_LIST) MEMALLOC(dwEntryListSize);
if ( !*ppEntryListHead )
{
ShowMessage(stderr,GetResString(IDS_ERROR_ENUMERATE));
status = STATUS_NO_MEMORY;
bNobreak = FALSE;
break;
}
status = NtEnumerateBootEntries(
(PVOID) *ppEntryListHead,
&dwEntryListSize
);
if ( !NT_SUCCESS(status) )
{
if ( *ppEntryListHead ) {
MEMFREE(*ppEntryListHead);
*ppEntryListHead = NULL;
}
if ( STATUS_INSUFFICIENT_RESOURCES == status ) {
dwEntryListSize += 0x0001000;
continue;
}
else
{
ShowMessage(stderr,GetResString(IDS_ERROR_ENUMERATE));
bNobreak = FALSE;
break;
}
}
else {
break;
}
}while (TRUE==bNobreak);
return status;
}
DWORD
GetDeviceInfo( IN LPTSTR szGUID,
OUT LPTSTR szFinalStr,
IN DWORD dwDriveNum,
IN DWORD dwActuals)
/*++
Routine description : This routine is used to retrieve the list of boot entries.
Arguments:
[ in ] szGUID : The address of a pointer to a BOOT_ENTRY_LIST struct.
[ out ] szFinalStr : The String containing the final ARG signature path.
[ in ] dwDriveNum : Specifies the drive number
[ in ] dwActuals : Specifies
Return Value : DWORD
Returns EXIT_SUCCESS if it is successful,
returnS EXIT_FAILURE otherwise.
--*/
{
HRESULT hr = S_OK;
HANDLE hDevice ;
BOOL bResult = FALSE ;
PPARTITION_INFORMATION_EX pInfo ;
PDRIVE_LAYOUT_INFORMATION_EX Drive ;
DWORD dwBytesCount = 0 ;
TCHAR szDriveName[MAX_RES_STRING+1] = NULL_STRING ;
DWORD dwErrCode = 0 ;
DWORD dwStructSize = 0 ;
TCHAR szInstallPath[MAX_RES_STRING+1] = NULL_STRING;
TCHAR szInstallPath1[MAX_RES_STRING*2] = NULL_STRING;
TCHAR szWindowsDirectory[MAX_PATH*2] = NULL_STRING;
TCHAR szMessage[MAX_RES_STRING+1] = NULL_STRING;
PTCHAR pszTok = NULL ;
PPARTITION_INFORMATION_GPT pGptPartition ;
UUID MyGuid ;
DWORD dwPartitionId = 0 ;
BOOL bGuidFlag = FALSE ;
BOOL bFoundFlag = TRUE ;
DWORD dwReqdSize = 0 ;
if ( UuidFromString(szGUID,&MyGuid) != RPC_S_OK )
{
ShowMessage(stderr,GetResString(IDS_INVALID_GUID));
return EXIT_FAILURE ;
}
while(bFoundFlag == TRUE)
{
SecureZeroMemory(szDriveName, SIZE_OF_ARRAY(szDriveName));
hr = StringCchPrintf(szDriveName, SIZE_OF_ARRAY(szDriveName), _T("\\\\.\\physicaldrive%d"), dwDriveNum );
hDevice = CreateFile(szDriveName,
GENERIC_READ|GENERIC_WRITE,
FILE_SHARE_READ|FILE_SHARE_WRITE,
NULL,
OPEN_EXISTING,
0,
NULL);
if(hDevice == INVALID_HANDLE_VALUE)
{
dwErrCode = GetLastError();
bFoundFlag =FALSE ;
// Display ann error message and exit if the user has mentioned
// any disk number.
if ( dwActuals == 1)
{
ShowMessage(stderr,GetResString(IDS_INVALID_DISK) );
return EXIT_FAILURE ;
}
else
{
break ;
}
}
//increase the drive number.
dwDriveNum++ ;
//Drive = (PDRIVE_LAYOUT_INFORMATION_EX)malloc(sizeof(DRIVE_LAYOUT_INFORMATION_EX) +5000) ;
dwReqdSize = sizeof(DRIVE_LAYOUT_INFORMATION_EX)+ sizeof(PARTITION_INFORMATION)*50 ;
Drive = (PDRIVE_LAYOUT_INFORMATION_EX)AllocateMemory(sizeof(DRIVE_LAYOUT_INFORMATION_EX) + sizeof(PARTITION_INFORMATION)*50) ;
if(Drive == NULL)
{
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_SYSTEM );
CloseHandle(hDevice );
return EXIT_FAILURE ;
}
dwStructSize = sizeof(DRIVE_LAYOUT_INFORMATION_EX) ;
bResult = DeviceIoControl(
hDevice,
IOCTL_DISK_GET_DRIVE_LAYOUT_EX,
NULL,
0,
Drive,
dwReqdSize,
&dwBytesCount,
NULL);
//Drive = realloc(Drive,malloc(sizeof(DRIVE_LAYOUT_INFORMATION_EX) )+500 ) ;
if(bResult ==0)
{
SAFEFREE(Drive);
dwErrCode = GetLastError();
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_SYSTEM );
CloseHandle(hDevice );
return EXIT_FAILURE ;
}
//get a pointer to the PARTITION_INFORMATION_EX structure
for(dwPartitionId = 0 ;dwPartitionId < Drive->PartitionCount ; dwPartitionId++)
{
//get a pointer to the corresponding partition.
pInfo = (PPARTITION_INFORMATION_EX)(&Drive->PartitionEntry[dwPartitionId] ) ;
//get a pointer to the PARTITION_INFORMATION_GPT structure.
pGptPartition = AllocateMemory( sizeof( PARTITION_INFORMATION_GPT));
if(pGptPartition == NULL)
{
SAFEFREE(Drive);
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_SYSTEM );
CloseHandle(hDevice );
return EXIT_FAILURE ;
}
CopyMemory(pGptPartition,&pInfo->Gpt,sizeof(PARTITION_INFORMATION_GPT) );
if( ( MyGuid.Data1 == pGptPartition->PartitionId.Data1 ) &&
( MyGuid.Data2 == pGptPartition->PartitionId.Data2 )
&& (MyGuid.Data3 == pGptPartition->PartitionId.Data3)
&& (MyGuid.Data4[0] == pGptPartition->PartitionId.Data4[0])
&& (MyGuid.Data4[1] == pGptPartition->PartitionId.Data4[1])
&& (MyGuid.Data4[2] == pGptPartition->PartitionId.Data4[2])
&&(MyGuid.Data4[3] == pGptPartition->PartitionId.Data4[3])
&& (MyGuid.Data4[4]== pGptPartition->PartitionId.Data4[4] )
&& (MyGuid.Data4[5]== pGptPartition->PartitionId.Data4[5] )
&& (MyGuid.Data4[6]== pGptPartition->PartitionId.Data4[6] )
&& (MyGuid.Data4[7]== pGptPartition->PartitionId.Data4[7] ) )
{
SecureZeroMemory(szMessage, SIZE_OF_ARRAY(szMessage));
hr = StringCchPrintf(szMessage, SIZE_OF_ARRAY(szMessage),GetResString(IDS_GUID_FOUND),dwPartitionId+1);
bGuidFlag = TRUE ;
bFoundFlag =FALSE ;
goto out ;
}
}
CloseHandle(hDevice );
}
if(bGuidFlag == FALSE )
{
SAFEFREE(Drive);
SAFEFREE(pGptPartition);
ShowMessage(stdout,GetResString(IDS_GUID_ABSENT));
CloseHandle(hDevice );
return EXIT_FAILURE ;
}
out: if( 0 == GetWindowsDirectory(szWindowsDirectory,MAX_PATH) )
{
SAFEFREE(Drive);
SAFEFREE(pGptPartition);
ShowMessage(stderr,GetResString(IDS_ERROR_DRIVE));
CloseHandle(hDevice );
return EXIT_FAILURE ;
}
StringConcat(szWindowsDirectory,_T("*"), SIZE_OF_ARRAY(szWindowsDirectory));
pszTok = _tcstok(szWindowsDirectory,_T("\\"));
if(pszTok == NULL)
{ SAFEFREE(Drive);
SAFEFREE(pGptPartition);
ShowMessage(stderr,GetResString(IDS_TOKEN_ABSENT));
CloseHandle(hDevice );
return EXIT_FAILURE ;
}
pszTok = _tcstok(NULL,_T("*"));
if(pszTok == NULL)
{
SAFEFREE(Drive);
SAFEFREE(pGptPartition);
ShowMessage(stderr,GetResString(IDS_TOKEN_ABSENT));
CloseHandle(hDevice );
return EXIT_FAILURE ;
}
//prints the path into the string.
hr = StringCchPrintf( szInstallPath, SIZE_OF_ARRAY(szInstallPath),
ARC_SIGNATURE,
pGptPartition->PartitionId.Data1,
pGptPartition->PartitionId.Data2,
pGptPartition->PartitionId.Data3,
pGptPartition->PartitionId.Data4[0],
pGptPartition->PartitionId.Data4[1],
pGptPartition->PartitionId.Data4[2],
pGptPartition->PartitionId.Data4[3],
pGptPartition->PartitionId.Data4[4],
pGptPartition->PartitionId.Data4[5],
pGptPartition->PartitionId.Data4[6],
pGptPartition->PartitionId.Data4[7],
dwPartitionId + 1 ,
pInfo->StartingOffset,
pInfo->PartitionLength
);
SecureZeroMemory(szInstallPath1, SIZE_OF_ARRAY(szInstallPath1) );
hr = StringCchPrintf( szInstallPath1, SIZE_OF_ARRAY(szInstallPath1), _T("%s\\%s"), szInstallPath, pszTok);
StringCopy(szFinalStr,szInstallPath, MAX_RES_STRING+1 );
SAFEFREE(Drive);
SAFEFREE(pGptPartition);
CloseHandle(hDevice );
return EXIT_SUCCESS ;
}
DWORD
ProcessListSwitch_IA64( IN DWORD argc,
IN LPCTSTR argv[]
)
/*++
Routine description : This routine is used to retrieve and display the list of boot entries.
Arguments:
argc : command line arguments count.
argv :
Return Value : DWORD
Returns EXIT_SUCCESS if it is successful,
returns EXIT_FAILURE otherwise.
--*/
{
BOOL bUsage = FALSE ;
BOOL bList = FALSE;
DWORD dwExitCode = ERROR_SUCCESS;
DWORD dwList = 0 ;
TCHAR szList[MAX_STRING_LENGTH] = NULL_STRING ;
LPTSTR pszStopStr = NULL;
DWORD dwExitcode = 0 ;
TCMDPARSER2 cmdOptions[3];
PTCMDPARSER2 pcmdOption;
SecureZeroMemory(cmdOptions, SIZE_OF_ARRAY(cmdOptions)*sizeof(TCMDPARSER2) );
//main option
pcmdOption = &cmdOptions[0];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_LIST;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->pValue = &bList;
//main option
pcmdOption = &cmdOptions[1];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_DEFAULT;
pcmdOption->dwType = CP_TYPE_TEXT;
pcmdOption->dwFlags = CP2_DEFAULT;
pcmdOption->dwCount = 1;
pcmdOption->pValue = szList;
pcmdOption->dwLength = MAX_STRING_LENGTH;
//id usage
pcmdOption = &cmdOptions[2];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_USAGE;
pcmdOption->dwFlags = CP2_USAGE;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bUsage;
// Parsing the copy option switches
if ( !(DoParseParam2( argc, argv, 0, SIZE_OF_ARRAY(cmdOptions ), cmdOptions, 0 ) ) )
{
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return (EXIT_FAILURE);
}
//check if usage is specified with more than one option
if( (TRUE == bUsage) && (argc > 3) )
{
ShowMessage(stderr,GetResString(IDS_LIST_SYNTAX));
return ( EXIT_FAILURE );
}
// Displaying query usage if user specified -? with -query option
if( bUsage )
{
displayListUsage_IA64() ;
return (EXIT_SUCCESS);
}
TrimString(szList,TRIM_ALL);
dwExitcode = InitializeEFI();
if(EXIT_FAILURE == dwExitcode )
{
return EXIT_FAILURE ;
}
//if empty value is specified
if( cmdOptions[1].dwActuals != 0 && StringLength(szList,0) == 0 )
{
ShowMessage(stderr,GetResString(IDS_LIST_SYNTAX));
return ( EXIT_FAILURE );
}
dwList = _tcstoul(szList,&pszStopStr, 10);
if ( StringLengthW(pszStopStr,0) != 0 )
{
ShowMessage(stderr,GetResString(IDS_INVALID_LISTVALUE));
return EXIT_FAILURE;
}
if(dwList > 0)
{
dwExitCode= ListDeviceInfo(dwList );
return (dwExitCode);
}
else
{
dwList = 0 ;
dwExitCode = ListDeviceInfo(dwList);
return (dwExitCode);
}
return EXIT_SUCCESS ;
}
VOID
displayListUsage_IA64()
/*++
Routine Description : Display the help for the list option (IA64).
Arguments :
: NONE
Return Type : VOID
--*/
{
DWORD dwIndex = IDS_LIST_BEGIN_IA64 ;
for(;dwIndex <=IDS_LIST_END_IA64 ;dwIndex++)
{
ShowMessage(stdout,GetResString(dwIndex));
}
}
VOID
displayUpdateUsage_IA64()
/*++
Routine Description : Display the help for the update option (IA64).
Arguments :
: NONE
Return Type : VOID
--*/
{
DWORD dwIndex = IDS_UPDATE_BEGIN_IA64 ;
for(;dwIndex <=IDS_UPDATE_END_IA64 ;dwIndex++)
{
ShowMessage(stdout,GetResString(dwIndex));
}
}
DWORD
ProcessUpdateSwitch_IA64( IN DWORD argc,
IN LPCTSTR argv[]
)
/*++
Routine Description : Allows the user to update the OS load options specifed
based on the plex
Arguments :
[ in ] argc - Number of command line arguments
[ in ] argv - Array containing command line arguments
Return Type : DWORD
--*/
{
BOOL bUsage = FALSE ;
TCHAR szUpdate[MAX_RES_STRING+1] = NULL_STRING ;
DWORD dwList = 0 ;
NTSTATUS status ;
TCHAR szFinalStr[MAX_RES_STRING+1] = NULL_STRING ;
TCHAR szBrackets[] = _T("{}");
PBOOT_ENTRY_LIST pEntryListHead = NULL;
PBOOT_ENTRY pTargetEntry = NULL;
DWORD dwActuals = 0 ;
DWORD dwExitcode = 0 ;
TCMDPARSER2 cmdOptions[2];
PTCMDPARSER2 pcmdOption;
SecureZeroMemory(cmdOptions, SIZE_OF_ARRAY(cmdOptions)*sizeof(TCMDPARSER2) );
//main option
pcmdOption = &cmdOptions[0];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_UPDATE;
pcmdOption->dwType = CP_TYPE_TEXT;
pcmdOption->dwFlags= CP2_VALUE_OPTIONAL;
pcmdOption->dwCount = 1;
pcmdOption->pValue = szUpdate;
pcmdOption->dwLength = MAX_STRING_LENGTH;
//id usage
pcmdOption = &cmdOptions[1];
StringCopyA( pcmdOption->szSignature, "PARSER2", 8 );
pcmdOption->pwszOptions = CMDOPTION_USAGE;
pcmdOption->dwFlags = CP2_USAGE;
pcmdOption->dwType = CP_TYPE_BOOLEAN;
pcmdOption->dwCount = 1;
pcmdOption->pValue = &bUsage;
// Parsing the copy option switches
if ( !(DoParseParam2( argc, argv, 0, SIZE_OF_ARRAY(cmdOptions ), cmdOptions, 0 ) ) )
{
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return (EXIT_FAILURE);
}
//check if usage is specified with more than one option
if( (TRUE == bUsage) && (argc > 3) )
{
ShowMessage(stderr,GetResString(IDS_UPDATE_SYNTAX));
return ( EXIT_FAILURE );
}
// Displaying query usage if user specified -? with -query option
if( bUsage )
{
displayUpdateUsage_IA64() ;
return (EXIT_SUCCESS);
}
dwExitcode = InitializeEFI();
if(EXIT_FAILURE == dwExitcode )
{
return EXIT_FAILURE ;
}
if(!bUsage && (StringLengthW(szUpdate,0) ==0) )
{
ShowMessage(stderr,GetResString(IDS_UPDATE_SYNTAX));
return EXIT_FAILURE ;
}
if(StringLengthW(szUpdate,0) !=0)
{
//
// Pass the GUID specified by the User
// and convert that into the ARC signature Path.
//
//Trim the Leading and trailing brackets specified
// by the user.
StrTrim(szUpdate, szBrackets);
//dwActuals = cmdOptions[2].dwActuals ;
dwActuals = 0 ;
if (GetDeviceInfo(szUpdate,szFinalStr,dwList,dwActuals) == EXIT_FAILURE )
{
return EXIT_FAILURE ;
}
//acquire the necessary privilages for querying and manipulating the NV RAM.
status = AcquirePrivilege( SE_SYSTEM_ENVIRONMENT_PRIVILEGE, TRUE );
if ( !NT_SUCCESS(status) )
{
ShowMessage(stderr,GetResString(IDS_INSUFF_PRIV));
return HRESULT_FROM_NT(status);
}
//Enumerate the list of Boot Entries in the NV Ram.
status = EnumerateBootEntries( (PVOID *) &pEntryListHead );
if ( !NT_SUCCESS(status) || !pEntryListHead )
{
if ( !pEntryListHead )
{
return EXIT_FAILURE ;
}
}
//
// Find The BootEntry corresponding to the ARC Signature path specified by the user.
//
//
status = FindBootEntry( pEntryListHead,szFinalStr,&pTargetEntry);
if ( !NT_SUCCESS(status) && STATUS_OBJECT_NAME_NOT_FOUND != status)
{
DISPLAY_MESSAGE(stderr,GetResString(IDS_FIND_FAIL) );
return EXIT_FAILURE ;
}
}
return EXIT_SUCCESS ;
}
DWORD
GetBootPath(IN LPTSTR szValue,
IN LPTSTR szResult
)
/*++
Routine Description : retreive the information from registry
Arguments :
[ in ] Keyname : System name
Return Type : DWORD
ERROR_SUCCESS : if successful in retreiving information.
ERROR_RETREIVE_REGISTRY : if error occurs while retreving information.
--*/
{
HKEY hKey1 = 0;
HKEY hRemoteKey = 0;
TCHAR szPath[MAX_STRING_LENGTH + 1] = SUBKEY1 ;
DWORD dwValueSize = MAX_STRING_LENGTH + 1;
DWORD dwRetCode = ERROR_SUCCESS;
DWORD dwError = 0;
TCHAR szTmpCompName[MAX_STRING_LENGTH+1] = NULL_STRING;
DWORD dwLength = MAX_STRING_LENGTH ;
LPTSTR szReturnValue = NULL ;
DWORD dwCode = 0 ;
szReturnValue = ( LPTSTR ) AllocateMemory( dwLength*sizeof( TCHAR ) );
if(szReturnValue == NULL)
{
return ERROR_RETREIVE_REGISTRY ;
}
// Get Remote computer local machine key
dwError = RegConnectRegistry(szTmpCompName,HKEY_LOCAL_MACHINE,&hRemoteKey);
if (dwError == ERROR_SUCCESS)
{
dwError = RegOpenKeyEx(hRemoteKey,szPath,0,KEY_READ,&hKey1);
if (dwError == ERROR_SUCCESS)
{
dwRetCode = RegQueryValueEx(hKey1, szValue, NULL, NULL,(LPBYTE) szReturnValue, &dwValueSize);
if (dwRetCode == ERROR_MORE_DATA)
{
if ( szReturnValue != NULL )
{
FreeMemory((LPVOID *) &szReturnValue );
szReturnValue = NULL;
}
szReturnValue = ( LPTSTR ) AllocateMemory( dwValueSize*sizeof( TCHAR ) );
if(szReturnValue == NULL)
{
RegCloseKey(hKey1);
RegCloseKey(hRemoteKey);
SAFEFREE(szReturnValue);
return ERROR_RETREIVE_REGISTRY ;
}
dwRetCode = RegQueryValueEx(hKey1, szValue, NULL, NULL,(LPBYTE) szReturnValue, &dwValueSize);
}
if(dwRetCode != ERROR_SUCCESS)
{
RegCloseKey(hKey1);
RegCloseKey(hRemoteKey);
SAFEFREE(szReturnValue);
return ERROR_RETREIVE_REGISTRY ;
}
}
else
{
RegCloseKey(hRemoteKey);
SAFEFREE(szReturnValue);
return ERROR_RETREIVE_REGISTRY ;
}
RegCloseKey(hKey1);
}
else
{
RegCloseKey(hRemoteKey);
SAFEFREE(szReturnValue);
return ERROR_RETREIVE_REGISTRY ;
}
RegCloseKey(hRemoteKey);
StringCopy(szResult,szReturnValue, MAX_RES_STRING+1);
SAFEFREE(szReturnValue);
return dwCode ;
}
NTSTATUS
LowGetPartitionInfo(
IN HANDLE handle,
OUT PARTITION_INFORMATION_EX *partitionData
)
/*++
Routine Description:
This routine gets the partition information given a handle to a partition.
Arguments:
handle - A handle to the partition.
partitionData - Returns a partition information structure.
Return Value:
Returns STATUS_SUCESS if successful, otherwise it returns the error code.
--*/
{
NTSTATUS status = STATUS_SUCCESS;
IO_STATUS_BLOCK statusBlock;
RtlZeroMemory( &statusBlock, sizeof(IO_STATUS_BLOCK) );
if ( (NULL == partitionData)
|| (sizeof(*partitionData) < sizeof(PARTITION_INFORMATION_EX))
|| (NULL == handle)
|| (INVALID_HANDLE_VALUE == handle) )
{
return STATUS_INVALID_PARAMETER;
}
RtlZeroMemory( partitionData, sizeof(PARTITION_INFORMATION_EX) );
status = NtDeviceIoControlFile(handle,
0,
NULL,
NULL,
&statusBlock,
IOCTL_DISK_GET_PARTITION_INFO_EX,
NULL,
0,
partitionData,
sizeof(PARTITION_INFORMATION_EX)
);
return status;
}
LONG
DmCommonNtOpenFile(
IN PWSTR Name,
IN ULONG access,
IN PHANDLE Handle
)
/*++
Routine Description:
This is a routine to handle open requests.
Arguments:
Name - pointer to the NT name for the open.
Handle - pointer for the handle returned.
Return Value:
NT status
--*/
{
OBJECT_ATTRIBUTES oa;
NTSTATUS status;
IO_STATUS_BLOCK statusBlock;
UNICODE_STRING unicodeName;
int i = 0 ;
status = RtlCreateUnicodeString(&unicodeName, Name);
if (!NT_SUCCESS(status))
{
return status;
}
RtlZeroMemory(&statusBlock, sizeof(IO_STATUS_BLOCK));
RtlZeroMemory(&oa, sizeof(OBJECT_ATTRIBUTES));
oa.Length = sizeof(OBJECT_ATTRIBUTES);
oa.ObjectName = &unicodeName;
oa.Attributes = OBJ_CASE_INSENSITIVE;
// If a sharing violation occurs,retry it for
// max. 10 seconds
for (i = 0; i < 5; i++)
{
status = NtOpenFile(Handle,
SYNCHRONIZE | access,
&oa,
&statusBlock,
FILE_SHARE_READ | FILE_SHARE_WRITE,
FILE_SYNCHRONOUS_IO_ALERT
);
if (status == STATUS_SHARING_VIOLATION) {
Sleep(2000);
}
else {
break;
}
}
RtlFreeUnicodeString(&unicodeName);
return status;
}
DWORD
AddMirrorPlex( IN LPTSTR szOsLoadPath ,
IN LPTSTR szLoaderPath ,
IN LPTSTR szValue ,
IN BOOL bFlag,
IN LPTSTR szFriendlyName
)
/*++
Routine Description:
This is a routine to Add a new mirror Entry
Arguments:
Return Value:
DWORD.
--*/
{
// local variables
HRESULT hr = S_OK;
BOOLEAN wasEnabled = TRUE;
DWORD dwAlign = 0;
DWORD dwError = 0;
DWORD dwLength = 0;
DWORD dwBootEntrySize = 0;
DWORD dwBootFilePathSize = 0;
DWORD dwOsLoadPathSize = 0;
DWORD dwWindowsOptionsSize = 0;
PBOOT_ENTRY pBootEntry = NULL;
PWINDOWS_OS_OPTIONS pWindowsOptions = NULL;
PFILE_PATH pBootFilePath = NULL;
PFILE_PATH pOsLoadPath = NULL;
ULONG* pdwIdsArray = NULL;
ULONG ulId = 0;
ULONG ulIdCount = 0;
NTSTATUS status;
TCHAR pwszBootFilePath[MAX_RES_STRING+1] = NULL_STRING;
PFILE_PATH pFilePath = NULL;
HANDLE hPart = INVALID_HANDLE_VALUE;
GUID guid;
PARTITION_INFORMATION_EX PartitionInfo;
TCHAR szBootPath[MAX_RES_STRING+1] = NULL_STRING;
// enable the privilege that is necessary to query/set NVRAM.
status = RtlAdjustPrivilege( SE_SYSTEM_ENVIRONMENT_PRIVILEGE, TRUE, FALSE, &wasEnabled );
if ( !NT_SUCCESS( status ) )
{
dwError = RtlNtStatusToDosError( status );
DISPLAY_MESSAGE( stderr, GetResString(IDS_INSUFF_PRIV));
return EXIT_FAILURE;
}
//
// open the system device
//
status = DmCommonNtOpenFile( szValue, GENERIC_READ, &hPart );
if ( status || !hPart || INVALID_HANDLE_VALUE == hPart )
{
dwError = RtlNtStatusToDosError( status );
DISPLAY_MESSAGE(stderr,GetResString(IDS_ERROR_ADD));
return EXIT_FAILURE;
}
//
// The structure is zero'ed in this call before retrieving the data.
//
status = LowGetPartitionInfo( hPart, &PartitionInfo );
if ( status )
{
dwError = RtlNtStatusToDosError( status );
DISPLAY_MESSAGE(stderr,GetResString(IDS_PARTITION_ERROR));
return EXIT_FAILURE;
}
if ( PARTITION_STYLE_GPT != PartitionInfo.PartitionStyle )
{
dwError = RtlNtStatusToDosError( status );
DISPLAY_MESSAGE(stderr,GetResString(IDS_ERROR_PARTITION_STYLE));
return EXIT_FAILURE;
}
//
// Setup the OSLoader file path.
//
guid = PartitionInfo.Gpt.PartitionId;
if (bFlag)
{
StringCopy(szBootPath,BOOTFILE_PATH, SIZE_OF_ARRAY(szBootPath));
}
else
{
StringCopy(szBootPath,BOOTFILE_PATH1, SIZE_OF_ARRAY(szBootPath) );
}
SecureZeroMemory(pwszBootFilePath, sizeof(pwszBootFilePath) );
hr = StringCchPrintf( pwszBootFilePath, SIZE_OF_ARRAY(pwszBootFilePath)-1,
szBootPath,
guid.Data1,
guid.Data2,
guid.Data3,
guid.Data4[0], guid.Data4[1], guid.Data4[2], guid.Data4[3],
guid.Data4[4], guid.Data4[5], guid.Data4[6], guid.Data4[7],
szLoaderPath);
//
// prepare the boot file path
//
//
// determine the length of the BOOTFILE_PATH
dwLength = StringLengthW( pwszBootFilePath,0) + 1;
// now determine the memory size that needs to be allocated for FILE_PATH structure
// and align up to the even memory bounday
dwBootFilePathSize = FIELD_OFFSET( FILE_PATH, FilePath ) + (dwLength * sizeof( WCHAR ));
// allocate the memory
pBootFilePath = (PFILE_PATH) AllocateMemory( sizeof( BYTE )*dwBootFilePathSize );
if ( NULL == pBootFilePath )
{
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL);
return EXIT_FAILURE;
}
// set the values now
SecureZeroMemory( pBootFilePath, dwBootFilePathSize ); // double init
pBootFilePath->Length = dwBootFilePathSize;
pBootFilePath->Type = FILE_PATH_TYPE_ARC_SIGNATURE;
pBootFilePath->Version = FILE_PATH_VERSION;
CopyMemory( pBootFilePath->FilePath, pwszBootFilePath, dwLength * sizeof( WCHAR ) );
//
// testing translating
//
pFilePath = (PFILE_PATH) AllocateMemory( sizeof( BYTE )* 1024 );
if(NULL == pFilePath )
{
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return EXIT_FAILURE;
}
ulId = 1024;
status = NtTranslateFilePath( pBootFilePath, FILE_PATH_TYPE_NT, pFilePath, &ulId );
if ( ! NT_SUCCESS( status ) )
{
dwError = RtlNtStatusToDosError( status );
DISPLAY_MESSAGE(stderr,GetResString(IDS_ERROR_ADD));
SAFEFREE( pBootFilePath );
SAFEFREE( pFilePath );
return EXIT_FAILURE;
}
//
// determine the length of the OSLOAD PATH
//
dwLength = StringLengthW( szOsLoadPath,0 ) + 1;
// now determine the memory size that needs to be allocated for FILE_PATH structure
// and align up to the even memory bounday
dwOsLoadPathSize = FIELD_OFFSET( FILE_PATH, FilePath ) + (dwLength * sizeof( WCHAR ));
// allocate the memory
pOsLoadPath = (PFILE_PATH) AllocateMemory( sizeof( BYTE )*dwOsLoadPathSize );
if(pOsLoadPath == NULL)
{
SAFEFREE( pBootFilePath );
SAFEFREE( pFilePath);
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
DISPLAY_MESSAGE( stderr, ERROR_TAG);
ShowLastError(stderr);
return EXIT_FAILURE;
}
// set the values now
SecureZeroMemory( pOsLoadPath, dwOsLoadPathSize ); // double init
pOsLoadPath->Length = dwOsLoadPathSize;
pOsLoadPath->Type = FILE_PATH_TYPE_ARC_SIGNATURE;
pOsLoadPath->Version = FILE_PATH_VERSION;
CopyMemory( pOsLoadPath->FilePath, szOsLoadPath, dwLength * sizeof( WCHAR ) );
//
// windows os options
//
// determine the size needed
dwLength = 1; // os load options is empty string
dwWindowsOptionsSize = sizeof(WINDOWS_OS_OPTIONS) +
dwOsLoadPathSize + sizeof(DWORD); // Need to align the FILE_PATH struct
// allocate the memory
pWindowsOptions = (PWINDOWS_OS_OPTIONS) AllocateMemory( dwWindowsOptionsSize*sizeof( BYTE ) );
if(pWindowsOptions == NULL)
{
SAFEFREE( pBootFilePath );
SAFEFREE( pFilePath);
SAFEFREE( pOsLoadPath);
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL);
return EXIT_FAILURE;
}
// set the values now
SecureZeroMemory( pWindowsOptions, dwWindowsOptionsSize ); // double init
CopyMemory( (BYTE*) pWindowsOptions->Signature, WINDOWS_OS_OPTIONS_SIGNATURE, sizeof(WINDOWS_OS_OPTIONS_SIGNATURE) );
pWindowsOptions->Length = dwWindowsOptionsSize;
pWindowsOptions->Version = WINDOWS_OS_OPTIONS_VERSION;
pWindowsOptions->OsLoadPathOffset = sizeof( WINDOWS_OS_OPTIONS );
//
// Need to align the OsLoadPathOffset on a 4 byte boundary.
//
dwAlign = ( pWindowsOptions->OsLoadPathOffset & (sizeof(DWORD) - 1) );
if ( dwAlign != 0 )
{
pWindowsOptions->OsLoadPathOffset += sizeof(DWORD) - dwAlign;
}
StringCopy(pWindowsOptions->OsLoadOptions, L"", StringLengthW(L"",0) );
CopyMemory( ((BYTE*) pWindowsOptions) + pWindowsOptions->OsLoadPathOffset, pOsLoadPath, dwOsLoadPathSize );
//
// prepare the boot entry
//
// find the length of the friendly name
dwLength = StringLengthW( szFriendlyName, 0 ) + 1;
// determine the size of the structure
dwBootEntrySize = FIELD_OFFSET( BOOT_ENTRY, OsOptions ) +
dwWindowsOptionsSize +
( dwLength * sizeof( WCHAR ) ) +
dwBootFilePathSize +
+ sizeof(WCHAR) // Need to align the FriendlyName on WCHAR
+ sizeof(DWORD); // Need to align the BootFilePath on DWORD
// allocate memory
pBootEntry = (PBOOT_ENTRY) AllocateMemory( sizeof( BYTE )*dwBootEntrySize );
if(pBootEntry == NULL)
{
SAFEFREE( pBootFilePath );
SAFEFREE( pFilePath);
SAFEFREE( pOsLoadPath);
SAFEFREE( pWindowsOptions);
ShowLastErrorEx(stderr, SLE_TYPE_ERROR|SLE_INTERNAL );
return EXIT_FAILURE;
}
// set the values now
SecureZeroMemory( pBootEntry, dwBootEntrySize );
pBootEntry->Version = BOOT_ENTRY_VERSION;
pBootEntry->Length = dwBootEntrySize;
pBootEntry->Id = 0L;
pBootEntry->Attributes = BOOT_ENTRY_ATTRIBUTE_ACTIVE;
pBootEntry->FriendlyNameOffset = FIELD_OFFSET(BOOT_ENTRY, OsOptions) + dwWindowsOptionsSize;
//
// Need to align the unicode string on a 2 byte boundary.
//
dwAlign = ( pBootEntry->FriendlyNameOffset & (sizeof(WCHAR) - 1) );
if ( dwAlign != 0 )
{
pBootEntry->FriendlyNameOffset += sizeof(WCHAR) - dwAlign;
}
pBootEntry->BootFilePathOffset = pBootEntry->FriendlyNameOffset + ( dwLength * sizeof(WCHAR) );
//
// Need to align the FILE_PATH struct on a 4 byte boundary.
//
dwAlign = ( pBootEntry->BootFilePathOffset & (sizeof(DWORD) - 1) );
if ( dwAlign != 0 )
{
pBootEntry->BootFilePathOffset += sizeof(DWORD) - dwAlign;
}
pBootEntry->OsOptionsLength = dwWindowsOptionsSize;
CopyMemory( pBootEntry->OsOptions, pWindowsOptions, dwWindowsOptionsSize );
CopyMemory( ((PBYTE) pBootEntry) + pBootEntry->FriendlyNameOffset, szFriendlyName, ( dwLength * sizeof(WCHAR) ) );
CopyMemory( ((PBYTE) pBootEntry) + pBootEntry->BootFilePathOffset, pBootFilePath, dwBootFilePathSize );
//
// add the prepared boot entry
//
status = NtAddBootEntry( pBootEntry, &ulId );
if ( ! NT_SUCCESS( status ) )
{
dwError = RtlNtStatusToDosError( status );
SAFEFREE( pBootFilePath );
SAFEFREE( pFilePath);
SAFEFREE( pOsLoadPath);
SAFEFREE( pWindowsOptions);
DISPLAY_MESSAGE(stderr,GetResString(IDS_ERROR_ADD));
return EXIT_FAILURE;
}
else
{
DISPLAY_MESSAGE(stdout,GetResString(IDS_MIRROR_ADDED));
}
//
// Add the entry to the boot order.
//
ulIdCount = 32L;
pdwIdsArray = (PULONG) AllocateMemory(ulIdCount * sizeof(ULONG));
if(!pdwIdsArray)
{
SAFEFREE( pBootFilePath );
SAFEFREE( pFilePath);
SAFEFREE( pOsLoadPath);
SAFEFREE( pWindowsOptions);
SAFEFREE( pBootEntry);
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return EXIT_FAILURE;
}
status = NtQueryBootEntryOrder( pdwIdsArray, &ulIdCount );
if (! NT_SUCCESS( status ) )
{
ShowMessage(stderr,GetResString(IDS_ERROR_SET_BOOTENTRY));
SAFEFREE( pBootFilePath );
SAFEFREE( pFilePath);
SAFEFREE( pOsLoadPath);
SAFEFREE( pWindowsOptions);
SAFEFREE( pBootEntry);
SAFEFREE( pdwIdsArray);
return EXIT_SUCCESS ;
}
//
// Need room in the buffer for the new entry.
//
if ( 31L < ulIdCount )
{
pdwIdsArray = (PULONG) AllocateMemory( (ulIdCount+1) * sizeof(ULONG));
if(!pdwIdsArray)
{
SAFEFREE( pBootFilePath );
SAFEFREE( pFilePath);
SAFEFREE( pOsLoadPath);
SAFEFREE( pWindowsOptions);
SAFEFREE( pBootEntry);
SAFEFREE( pdwIdsArray);
ShowLastErrorEx(stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
return EXIT_FAILURE;
}
status = NtQueryBootEntryOrder( pdwIdsArray, &ulIdCount );
}
if ( !NT_SUCCESS(status) )
{
SAFEFREE( pBootFilePath );
SAFEFREE( pFilePath);
SAFEFREE( pOsLoadPath);
SAFEFREE( pWindowsOptions);
SAFEFREE( pBootEntry);
SAFEFREE( pdwIdsArray);
dwError = RtlNtStatusToDosError( status );
ShowMessage(stderr,GetResString(IDS_ERROR_ADD));
return EXIT_FAILURE;
}
ulIdCount++;
*(pdwIdsArray + (ulIdCount - 1)) = ulId;
status = NtSetBootEntryOrder( pdwIdsArray, ulIdCount );
if ( !NT_SUCCESS(status) )
{
SAFEFREE( pBootFilePath );
SAFEFREE( pFilePath);
SAFEFREE( pOsLoadPath);
SAFEFREE( pWindowsOptions);
SAFEFREE( pBootEntry);
SAFEFREE( pdwIdsArray);
dwError = RtlNtStatusToDosError( status );
ShowMessage(stderr,GetResString(IDS_ERROR_ADD));
return EXIT_FAILURE;
}
//
// release the allocated memory
//
SAFEFREE( pBootFilePath );
SAFEFREE( pFilePath);
SAFEFREE( pOsLoadPath);
SAFEFREE( pWindowsOptions);
SAFEFREE( pBootEntry);
SAFEFREE( pdwIdsArray);
return EXIT_SUCCESS;
}
DWORD
ConvertintoLocale( IN LPWSTR szTempBuf,
OUT LPWSTR szOutputStr )
/*++
Routine Description:
Converts into Locale and Gets the Locale information
Arguments:
LPWSTR szTempBuf [in] -- Locale Information to get
LPWSTR szOutputStr [out] -- Locale value corresponding to the given
information
Return Value:
DWORD
--*/
{
NUMBERFMT numberfmt;
WCHAR szGrouping[MAX_RES_STRING+1] = NULL_STRING;
WCHAR szDecimalSep[MAX_RES_STRING+1] = NULL_STRING;
WCHAR szThousandSep[MAX_RES_STRING+1] = NULL_STRING;
WCHAR szTemp[MAX_RES_STRING+1] = NULL_STRING;
LPWSTR szTemp1 = NULL;
LPWSTR pszStoppedString = NULL;
DWORD dwStatus = 0;
DWORD dwGrouping = 0;
//make the fractional digits and leading zeros to nothing
numberfmt.NumDigits = 0;
numberfmt.LeadingZero = 0;
//get the decimal seperate character
if(GetLocaleInfo( LOCALE_USER_DEFAULT, LOCALE_SDECIMAL, szDecimalSep, SIZE_OF_ARRAY(szDecimalSep) ) == 0)
{
return EXIT_FAILURE;
}
numberfmt.lpDecimalSep = szDecimalSep;
if(GetLocaleInfo( LOCALE_USER_DEFAULT, LOCALE_STHOUSAND, szThousandSep, SIZE_OF_ARRAY(szThousandSep) ) == 0)
{
return EXIT_FAILURE;
}
numberfmt.lpThousandSep = szThousandSep;
if(GetLocaleInfo( LOCALE_USER_DEFAULT, LOCALE_SGROUPING, szGrouping, SIZE_OF_ARRAY(szGrouping) ) == 0)
{
return EXIT_FAILURE;
}
szTemp1 = wcstok( szGrouping, L";");
do
{
StringConcat( szTemp, szTemp1, SIZE_OF_ARRAY(szTemp));
szTemp1 = wcstok( NULL, L";" );
}while( szTemp1 != NULL && StringCompare( szTemp1, L"0", TRUE, 0) != 0);
dwGrouping = wcstol( szTemp, &pszStoppedString, 10);
numberfmt.Grouping = (UINT)dwGrouping ;
numberfmt.NegativeOrder = 2;
dwStatus = GetNumberFormat( LOCALE_USER_DEFAULT, 0, szTempBuf, &numberfmt, szOutputStr, MAX_RES_STRING+1);
return(EXIT_SUCCESS);
}
void Freelist()
/*++
Routine Description : Function used to free the global linked list
Arguments:
Return Type :
--*/
{
PLIST_ENTRY listEntry;
PLIST_ENTRY listEntry1;
PMY_BOOT_ENTRY bootEntry;
listEntry = BootEntries.Flink;
while( listEntry != &BootEntries)
{
listEntry1 = listEntry;
bootEntry = CONTAINING_RECORD( listEntry, MY_BOOT_ENTRY, ListEntry );
RemoveEntryList( &bootEntry->ListEntry );
listEntry = listEntry->Flink;
if( listEntry1 != NULL )
{
FreeMemory((LPVOID *) &listEntry1 );
}
}
}
PWSTR GetDefaultBootEntry()
/*++
Routine Description :
Gets the default Boot entry.
Arguments :
Return Type : PWSTR
Returns the first entry in the list.
--*/
{
PLIST_ENTRY listEntry;
PMY_BOOT_ENTRY bootEntry;
PWSTR NtFilePath=NULL;
listEntry = BootEntries.Flink;
bootEntry = CONTAINING_RECORD( listEntry, MY_BOOT_ENTRY, ListEntry );
NtFilePath = GetNtNameForFilePath(bootEntry->OsFilePath);
return (NtFilePath);
}