|
|
/*++
Copyright (c) 1993 Microsoft Corporation
Module Name:
arc.c
Abstract:
ARC/NV-RAM manipulation routines for 32-bit winnt setup.
Author:
Ted Miller (tedm) 19-December-1993
Revision History:
--*/
#include "precomp.h"
#pragma hdrstop
#include "initguid.h"
#include "diskguid.h"
#ifdef UNICODE // Always true for ARC, never true for Win9x upgrade
#if defined(_X86_)
LOGICAL IsArcChecked = FALSE; LOGICAL IsArcMachine; #endif
#if defined(EFI_NVRAM_ENABLED)
#include <ntosp.h> // for ALIGN_UP
LOGICAL IsEfiChecked = FALSE; LOGICAL IsEfiMachine;
DWORD InitializeEfiStuff( IN HWND Parent );
NTSTATUS (*AddBootEntry) ( IN PBOOT_ENTRY BootEntry, OUT PULONG Id OPTIONAL );
NTSTATUS (*DeleteBootEntry) ( IN ULONG Id );
NTSTATUS (*EnumerateBootEntries) ( OUT PVOID Buffer, IN OUT PULONG BufferLength );
NTSTATUS (*QueryBootEntryOrder) ( OUT PULONG Ids, IN OUT PULONG Count );
NTSTATUS (*SetBootEntryOrder) ( IN PULONG Ids, IN ULONG Count );
NTSTATUS (*QueryBootOptions) ( OUT PBOOT_OPTIONS BootOptions, IN OUT PULONG BootOptionsLength );
NTSTATUS (*SetBootOptions) ( IN PBOOT_OPTIONS BootOptions, IN ULONG FieldsToChange );
PBOOT_OPTIONS BootOptions = NULL; PBOOT_OPTIONS OriginalBootOptions = NULL; PULONG OriginalBootEntryOrder = NULL; ULONG OriginalBootEntryOrderCount; PBOOT_ENTRY_LIST BootEntries = NULL;
//
// MY_BOOT_ENTRY is the internal representation of an EFI NVRAM boot item.
// The NtBootEntry item is the structure passed to/from the NT boot entry APIs.
//
typedef struct _MY_BOOT_ENTRY { struct _MY_BOOT_ENTRY *Next; PUCHAR AllocationEnd; ULONG Status; PWSTR FriendlyName; ULONG FriendlyNameLength; PWSTR OsLoadOptions; ULONG OsLoadOptionsLength; PFILE_PATH BootFilePath; PFILE_PATH OsFilePath; BOOT_ENTRY NtBootEntry; } MY_BOOT_ENTRY, *PMY_BOOT_ENTRY;
#define MBE_STATUS_ORDERED 0x00000001
#define MBE_STATUS_NEW 0x00000002
#define MBE_STATUS_DELETED 0x00000004
#define MBE_STATUS_COMMITTED 0x00000008
#define IS_BOOT_ENTRY_ACTIVE(_be) \
(((_be)->NtBootEntry.Attributes & BOOT_ENTRY_ATTRIBUTE_ACTIVE) != 0) #define IS_BOOT_ENTRY_WINDOWS(_be) \
(((_be)->NtBootEntry.Attributes & BOOT_ENTRY_ATTRIBUTE_WINDOWS) != 0) #define IS_BOOT_ENTRY_REMOVABLE_MEDIA(_be) \
(((_be)->NtBootEntry.Attributes & BOOT_ENTRY_ATTRIBUTE_REMOVABLE_MEDIA) != 0)
#define IS_BOOT_ENTRY_ORDERED(_be) \
(((_be)->Status & MBE_STATUS_ORDERED) != 0) #define IS_BOOT_ENTRY_NEW(_be) \
(((_be)->Status & MBE_STATUS_NEW) != 0) #define IS_BOOT_ENTRY_DELETED(_be) \
(((_be)->Status & MBE_STATUS_DELETED) != 0) #define IS_BOOT_ENTRY_COMMITTED(_be) \
(((_be)->Status & MBE_STATUS_COMMITTED) != 0)
PMY_BOOT_ENTRY MyBootEntries = NULL;
NTSTATUS ConvertBootEntries( VOID );
BOOL CreateBootEntry( PWSTR BootFileDevice, PWSTR BootFilePath, PWSTR OsLoadDevice, PWSTR OsLoadPath, PWSTR OsLoadOptions, PWSTR FriendlyName );
#define ADD_OFFSET(_p,_o) (PVOID)((PUCHAR)(_p) + (_p)->_o)
#endif // defined(EFI_NVRAM_ENABLED)
UINT SystemPartitionCount; PWSTR* SystemPartitionNtNames; PWSTR SystemPartitionNtName; PWSTR SystemPartitionVolumeGuid;
typedef enum { BootVarSystemPartition, BootVarOsLoader, BootVarOsLoadPartition, BootVarOsLoadFilename, BootVarLoadIdentifier, BootVarOsLoadOptions, BootVarMax } BootVars;
LPCWSTR BootVarNames[BootVarMax] = { L"SYSTEMPARTITION", L"OSLOADER", L"OSLOADPARTITION", L"OSLOADFILENAME", L"LOADIDENTIFIER", L"OSLOADOPTIONS" };
LPCWSTR szAUTOLOAD = L"AUTOLOAD"; LPCWSTR szCOUNTDOWN = L"COUNTDOWN";
LPWSTR BootVarValues[BootVarMax];
LPCWSTR OriginalBootVarValues[BootVarMax];
LPCWSTR OriginalCountdown; LPCWSTR OriginalAutoload;
DWORD BootVarComponentCount[BootVarMax]; LPWSTR *BootVarComponents[BootVarMax]; DWORD LargestComponentCount;
LPWSTR DosDeviceTargets[26];
//
// Flag indicating whether we messed with NV-RAM and thus need to
// try to restore it in case the user cancels.
//
BOOL CleanUpNvRam;
//
// Leave as array because some code uses sizeof(ArcNameDirectory)
//
WCHAR ArcNameDirectory[] = L"\\ArcName";
#define GLOBAL_ROOT L"\\\\?\\GLOBALROOT"
#define MAX_COMPONENTS 20
WCHAR ForcedSystemPartition;
//
// Helper macro to make object attribute initialization a little cleaner.
//
#define INIT_OBJA(Obja,UnicodeString,UnicodeText) \
\ RtlInitUnicodeString((UnicodeString),(UnicodeText)); \ \ InitializeObjectAttributes( \ (Obja), \ (UnicodeString), \ OBJ_CASE_INSENSITIVE, \ NULL, \ NULL \ )
UINT NormalizeArcPath( IN PCWSTR Path, OUT LPWSTR *NormalizedPath )
/*++
Routine Description:
Transform an ARC path into one with no sets of empty parenthesis (ie, transforom all instances of () to (0).).
Arguments:
Path - ARC path to be normalized.
NormalizedPath - if successful, receives a pointer to the normalized arc path. The caller must free with FREE().
Return Value:
Win32 error code indicating outcome.
--*/
{ LPWSTR r; LPCWSTR p,q; LPWSTR normalizedPath; UINT numEmpties=0; //Number of instances of "()" seen so far
UINT numEmptiesAllocated=100; //Number of instances of "()" accounted for
if(normalizedPath = MALLOC((lstrlen(Path)+1+numEmptiesAllocated)*sizeof(WCHAR))) { ZeroMemory(normalizedPath,(lstrlen(Path)+1+numEmptiesAllocated)*sizeof(WCHAR)); } else { return(ERROR_NOT_ENOUGH_MEMORY); }
for(p=Path; q=wcsstr(p,L"()"); p=q+2) {
numEmpties++; if (numEmpties > numEmptiesAllocated) { numEmptiesAllocated += 100;
r = REALLOC(normalizedPath,(lstrlen(Path)+1+numEmptiesAllocated)*sizeof(WCHAR));
if (r) { normalizedPath = r; } else { FREE(normalizedPath); return(ERROR_NOT_ENOUGH_MEMORY); } }
r = normalizedPath + lstrlen(normalizedPath); lstrcpyn(r,p,(int)(q-p)+1); lstrcat(normalizedPath,L"(0)"); } lstrcat(normalizedPath,p);
if(r = REALLOC(normalizedPath,(lstrlen(normalizedPath)+1)*sizeof(WCHAR))) { *NormalizedPath = r; return(NO_ERROR); } else { FREE(normalizedPath); return(ERROR_NOT_ENOUGH_MEMORY); } }
//
// ISSUE -- Since only MAX_COMPONENTS components are supported in the input string, is this
// a security hole, since having a boot.ini with more than 20 arc lines is not supported?
//
DWORD GetVarComponents( IN PCWSTR VarValue, OUT LPWSTR **Components, OUT PDWORD ComponentCount )
/*++
Routine Description:
Split a semi-colon delineated list of arc paths up into a set of individual strings. For each component leading and trailing spaces are stripped out.
Arguments:
VarValue - supplies string with list of arc paths to be split apart. Only a maximum of MAX_COMPONENTS components are supported.
Components - receives array of pointers to individual components on the variable specified in VarValue.
ComponentCount - receives number of separate arc paths in the Components array.
Return Value:
Win32 error indicating outcome. If NO_ERROR then the caller must free the Components array and the strings pointed to by its elements.
--*/
{ LPWSTR *components; LPWSTR *temp; DWORD componentCount; LPCWSTR p; LPCWSTR Var; LPWSTR comp; DWORD len; UINT ec;
components = MALLOC(MAX_COMPONENTS * sizeof(LPWSTR)); if(!components) { return(ERROR_NOT_ENOUGH_MEMORY); } ZeroMemory(components,MAX_COMPONENTS * sizeof(LPWSTR));
ec = NO_ERROR;
for(Var=VarValue,componentCount=0; *Var; ) {
//
// Skip leading spaces.
//
while((*Var == L' ') || (*Var == L'\t')) { Var++; }
if(*Var == 0) { break; }
p = Var;
while(*p && (*p != L';')) { p++; }
len = (DWORD)((PUCHAR)p - (PUCHAR)Var);
comp = MALLOC(len + sizeof(WCHAR)); if(!comp) { ec = ERROR_NOT_ENOUGH_MEMORY; break; }
len /= sizeof(WCHAR);
lstrcpynW(comp,Var,len+1);
ec = NormalizeArcPath(comp,&components[componentCount]); FREE(comp); if(ec != NO_ERROR) { break; }
componentCount++;
if(componentCount == MAX_COMPONENTS) { break; }
Var = p; if(*Var) { Var++; // skip ;
} }
if(ec == NO_ERROR) { if(componentCount) { temp = REALLOC(components,componentCount*sizeof(LPWSTR)); if(!temp) { ec = ERROR_NOT_ENOUGH_MEMORY; } } else { temp = NULL; } }
if(ec == NO_ERROR) { *Components = temp; *ComponentCount = componentCount; } else { for(len=0; components[len] && (len<MAX_COMPONENTS); len++) { FREE(components[len]); } FREE(components); }
return(ec); }
NTSTATUS QueryCanonicalName( IN PWSTR Name, IN ULONG MaxDepth, OUT PWSTR CanonicalName, IN ULONG SizeOfBufferInBytes ) /*++
Routine Description:
Resolves the symbolic name to the specified depth. To resolve a symbolic name completely specify the MaxDepth as -1
Arguments:
Name - Symbolic name to be resolved
MaxDepth - The depth till which the resolution needs to be carried out
CanonicalName - The fully resolved name
SizeOfBufferInBytes - The size of the CanonicalName buffer in bytes
Return Value:
Appropriate NT status code
--*/ { UNICODE_STRING name, canonName; OBJECT_ATTRIBUTES oa; NTSTATUS status; HANDLE handle; ULONG CurrentDepth;
RtlInitUnicodeString(&name, Name);
canonName.MaximumLength = (USHORT) (SizeOfBufferInBytes - sizeof(WCHAR)); canonName.Length = 0; canonName.Buffer = CanonicalName;
if (name.Length >= canonName.MaximumLength) { return STATUS_BUFFER_TOO_SMALL; }
RtlCopyMemory(canonName.Buffer, name.Buffer, name.Length); canonName.Length = name.Length; canonName.Buffer[canonName.Length/sizeof(WCHAR)] = 0;
for (CurrentDepth = 0; CurrentDepth < MaxDepth; CurrentDepth++) {
InitializeObjectAttributes(&oa, &canonName, OBJ_CASE_INSENSITIVE, 0, 0);
status = NtOpenSymbolicLinkObject(&handle, READ_CONTROL | SYMBOLIC_LINK_QUERY, &oa); if (!NT_SUCCESS(status)) { break; }
status = NtQuerySymbolicLinkObject(handle, &canonName, NULL); NtClose(handle);
if (!NT_SUCCESS(status)) { return status; }
canonName.Buffer[canonName.Length/sizeof(WCHAR)] = 0; }
return STATUS_SUCCESS; }
//
// Structure to map from old NT partition names like
// \device\harddisk0\partition1 to new NT partition names
// like \device\harddiskvolume1
//
typedef struct _NAME_TRANSLATIONS { WCHAR OldNtName[MAX_PATH]; WCHAR NewNtName[MAX_PATH]; } NT_NAME_TRANSLATION, * PNT_NAME_TRANSLATION;
//
// Map of old style NT partition names to new style NT
// partition names
//
NT_NAME_TRANSLATION OldNewNtNames[256] = {0};
PWSTR OldNtNameToNewNtName( IN PCWSTR OldNtName ) /*++
Routine Description:
Given a old format NT name tries to lookup at new format NT name in the global map
Arguments:
OldNtName - The partition name specified in the old format
Return Value:
The new NT name if there exists one, otherwise NULL.
--*/
{ ULONG Index = 0; ULONG MaxEntries = sizeof(OldNewNtNames)/sizeof(NT_NAME_TRANSLATION); PWSTR NewNtName = NULL;
for (Index = 0; (Index < MaxEntries); Index++) { if (OldNewNtNames[Index].OldNtName[0] && !_wcsicmp(OldNewNtNames[Index].OldNtName, OldNtName)) { NewNtName = OldNewNtNames[Index].NewNtName; } }
return NewNtName; }
PWSTR NewNtNameToOldNtName( IN PCWSTR NewNtName ) /*++
Routine Description:
Given a new format NT name tries to lookup at old format NT name in the global map
Arguments:
NewNtName - The partition name specified in the new format
Return Value:
The old NT name if there exists one, otherwise NULL.
--*/ { ULONG Index = 0; ULONG MaxEntries = sizeof(OldNewNtNames)/sizeof(NT_NAME_TRANSLATION); PWSTR OldNtName = NULL;
for (Index=0; (Index < MaxEntries); Index++) { if (OldNewNtNames[Index].NewNtName[0] && !_wcsicmp(OldNewNtNames[Index].NewNtName, NewNtName)) { OldNtName = OldNewNtNames[Index].OldNtName; } }
return OldNtName; }
DWORD InitOldToNewNtNameTranslations( VOID ) /*++
Routine Description:
Initializes the global old NT partition names to new NT partition names mapping.
Arguments:
None.
Return Value:
The number of valid entries in the map
--*/
{ DWORD MappingCount = 0; SYSTEM_DEVICE_INFORMATION SysDevInfo = {0}; NTSTATUS Status; OBJECT_ATTRIBUTES ObjAttrs; UNICODE_STRING ObjName;
Status = NtQuerySystemInformation(SystemDeviceInformation, &SysDevInfo, sizeof(SYSTEM_DEVICE_INFORMATION), NULL);
if (NT_SUCCESS(Status)) { ULONG Index; WCHAR OldNtPath[MAX_PATH]; DWORD ErrorCode = 0; ULONG SlotIndex = 0; ULONG MaxSlots = sizeof(OldNewNtNames)/sizeof(NT_NAME_TRANSLATION);
for (Index=0; (!ErrorCode) && (Index < SysDevInfo.NumberOfDisks) && (SlotIndex < MaxSlots); Index++) {
HANDLE DirectoryHandle;
swprintf(OldNtPath, L"\\device\\Harddisk%d", Index);
//
// Open the disk directory.
//
INIT_OBJA(&ObjAttrs, &ObjName, OldNtPath);
Status = NtOpenDirectoryObject(&DirectoryHandle, DIRECTORY_QUERY, &ObjAttrs);
if(NT_SUCCESS(Status)) { BOOLEAN RestartScan = TRUE; ULONG Context = 0; BOOLEAN MoreEntries = TRUE; WCHAR Buffer[MAX_PATH * 2] = {0}; POBJECT_DIRECTORY_INFORMATION DirInfo = (POBJECT_DIRECTORY_INFORMATION)Buffer;
do { Status = NtQueryDirectoryObject( DirectoryHandle, Buffer, sizeof(Buffer), TRUE, // return single entry
RestartScan, &Context, NULL // return length
);
if(NT_SUCCESS(Status)) { //
// Make sure this name is a symbolic link.
//
if(DirInfo->Name.Length && (DirInfo->TypeName.Length >= 24) && CharUpperBuff((LPWSTR)DirInfo->TypeName.Buffer,12) && !memcmp(DirInfo->TypeName.Buffer,L"SYMBOLICLINK",24)) { WCHAR EntryName[MAX_PATH];
StringCchCopy(EntryName, ARRAYSIZE(EntryName), OldNtPath);
ConcatenatePaths(EntryName, DirInfo->Name.Buffer, ARRAYSIZE(EntryName));
Status = QueryCanonicalName(EntryName, -1, Buffer, sizeof(Buffer));
if (NT_SUCCESS(Status)) {
StringCchCopyW( OldNewNtNames[SlotIndex].OldNtName, ARRAYSIZE(OldNewNtNames[0].OldNtName), EntryName);
StringCchCopyW( OldNewNtNames[SlotIndex].NewNtName, ARRAYSIZE(OldNewNtNames[0].NewNtName), Buffer);
SlotIndex++; } } } else { MoreEntries = FALSE;
if(Status == STATUS_NO_MORE_ENTRIES) { Status = STATUS_SUCCESS; }
ErrorCode = RtlNtStatusToDosError(Status); }
RestartScan = FALSE;
} while(MoreEntries && (SlotIndex < MaxSlots));
NtClose(DirectoryHandle); } else { ErrorCode = RtlNtStatusToDosError(Status); } }
if (!ErrorCode && NT_SUCCESS(Status)) { MappingCount = SlotIndex; } }
return MappingCount; }
DWORD NtNameToArcPath ( IN PCWSTR NtName, OUT LPWSTR *ArcPath )
/*++
Routine Description:
Convert an NT volume name to an ARC path.
Arguments:
NtName - supplies name of drive to be converted.
ArcPath - receives pointer to buffer containing arc path if the routine is successful. Caller must free with FREE().
Return Value:
Win32 error code indicating outcome.
--*/
{ UNICODE_STRING UnicodeString; HANDLE DirectoryHandle; OBJECT_ATTRIBUTES Obja; NTSTATUS Status; BOOLEAN RestartScan; DWORD Context; BOOL MoreEntries; LPWSTR ArcName = NULL; WCHAR Buffer[2 * MAX_PATH]; WCHAR ArcDiskName[MAX_PATH] = {0}; WCHAR NtDiskName[MAX_PATH] = {0}; WCHAR ArcPartitionName[MAX_PATH] = {0}; PWSTR PartitionName = NULL; PWSTR PartitionNumStr = NULL; POBJECT_DIRECTORY_INFORMATION DirInfo = (POBJECT_DIRECTORY_INFORMATION)Buffer; DWORD ErrorCode;
ErrorCode = NO_ERROR; *ArcPath = NULL;
//
// Get hold of the NT disk name
//
PartitionName = NewNtNameToOldNtName(NtName);
if (PartitionName) { PWSTR PartitionNameStart = PartitionName;
PartitionName = wcsrchr(PartitionName, L'\\');
if (PartitionName && wcsstr(PartitionName, L"Partition")) { wcsncpy(NtDiskName, PartitionNameStart, PartitionName - PartitionNameStart); wcscat(NtDiskName, L"\\Partition0"); PartitionNumStr = PartitionName + wcslen(L"\\Partition"); } }
//
// Open the \ArcName directory.
//
INIT_OBJA(&Obja,&UnicodeString,ArcNameDirectory);
Status = NtOpenDirectoryObject(&DirectoryHandle,DIRECTORY_QUERY,&Obja);
if(NT_SUCCESS(Status)) {
RestartScan = TRUE; Context = 0; MoreEntries = TRUE;
do {
Status = NtQueryDirectoryObject( DirectoryHandle, Buffer, sizeof(Buffer), TRUE, // return single entry
RestartScan, &Context, NULL // return length
);
if(NT_SUCCESS(Status)) {
CharLower(DirInfo->Name.Buffer);
//
// Make sure this name is a symbolic link.
//
if(DirInfo->Name.Length && (DirInfo->TypeName.Length >= 24) && CharUpperBuff((LPWSTR)DirInfo->TypeName.Buffer,12) && !memcmp(DirInfo->TypeName.Buffer,L"SYMBOLICLINK",24)) { WCHAR OldNtName[2 * MAX_PATH] = {0}; DWORD size = DirInfo->Name.Length + sizeof(ArcNameDirectory) + sizeof(WCHAR);
ArcName = MALLOC(size);
if(!ArcName) { ErrorCode = ERROR_NOT_ENOUGH_MEMORY;
break; }
//
// These two operations are safe, since we have accounted for the size of an
// ARC path, which could theoretically be larger than MAX_PATH
//
lstrcpy(ArcName,ArcNameDirectory); ConcatenatePaths(ArcName,DirInfo->Name.Buffer,size / sizeof(WCHAR));
//
// We have the entire arc name in ArcName. Now open the first
// level symbolic link.
//
Status = QueryCanonicalName(ArcName, 1, Buffer, sizeof(Buffer));
if (NT_SUCCESS(Status)) { wcscpy(OldNtName, Buffer);
//
// Now resolve the complete symbolic link
//
Status = QueryCanonicalName(ArcName, -1, Buffer, sizeof(Buffer));
if (NT_SUCCESS(Status)) { if(!lstrcmpi(Buffer, NtName)) { *ArcPath = ArcName + (sizeof(ArcNameDirectory)/sizeof(WCHAR)); } else { if (!lstrcmpi(OldNtName, NtDiskName)) { wcscpy(ArcDiskName, ArcName + (sizeof(ArcNameDirectory)/sizeof(WCHAR))); } } } else { if(!lstrcmpi(OldNtName, NtName)) { *ArcPath = ArcName + (sizeof(ArcNameDirectory)/sizeof(WCHAR)); } } }
if(!(*ArcPath)) { FREE(ArcName); ArcName = NULL; } } } else {
MoreEntries = FALSE;
if(Status == STATUS_NO_MORE_ENTRIES) { Status = STATUS_SUCCESS; }
ErrorCode = RtlNtStatusToDosError(Status); }
RestartScan = FALSE;
} while(MoreEntries && !(*ArcPath));
NtClose(DirectoryHandle); } else { ErrorCode = RtlNtStatusToDosError(Status); }
//
// If we found a match for the disk but not for the actual
// partition specified then guess thepartition number
// (based on the current nt partition number )
//
if ((!*ArcPath) && ArcDiskName[0] && PartitionName && PartitionNumStr) { PWSTR EndPtr = NULL; ULONG PartitionNumber = wcstoul(PartitionNumStr, &EndPtr, 10);
if (PartitionNumber) { StringCchPrintfW(ArcPartitionName, ARRAYSIZE(ArcPartitionName), L"%wspartition(%d)", ArcDiskName, PartitionNumber);
*ArcPath = DupString(ArcPartitionName); ErrorCode = NO_ERROR;
DebugLog( Winnt32LogInformation, TEXT("\nCould not find arcname mapping for %1 partition.\r\n") TEXT("Guessing the arcname to be %2"), 0, NtName, ArcPartitionName); } }
if (ErrorCode == NO_ERROR) { if(*ArcPath) { //
// ArcPath points into the middle of a buffer.
// The caller needs to be able to free it, so place it in its
// own buffer here.
//
*ArcPath = DupString(*ArcPath);
if (ArcName) { FREE(ArcName); }
if(*ArcPath == NULL) { ErrorCode = ERROR_NOT_ENOUGH_MEMORY; } } else { //
// No matching drive.
//
ErrorCode = ERROR_INVALID_DRIVE; } }
return ErrorCode; }
DWORD DriveLetterToArcPath( IN WCHAR DriveLetter, OUT LPWSTR *ArcPath )
/*++
Routine Description:
Convert a drive letter to an ARC path.
This routine relies on the DosDeviceTargets array being set up beforehand.
Arguments:
DriveLetter - supplies letter of drive to be converted.
ArcPath - receives pointer to buffer containing arc path if the routine is successful. Caller must free with FREE().
Return Value:
Win32 error code indicating outcome.
--*/
{ LPWSTR NtPath;
NtPath = DosDeviceTargets[(WCHAR)CharUpper((PWCHAR)DriveLetter)-L'A']; if(!NtPath) { return(ERROR_INVALID_DRIVE); }
return NtNameToArcPath (NtPath, ArcPath); }
DWORD ArcPathToDriveLetterAndNtName ( IN PCWSTR ArcPath, OUT PWCHAR DriveLetter, OUT PWSTR NtName, IN DWORD BufferSizeInBytes )
/*++
Routine Description:
Convert an arc path to a drive letter.
This routine relies on the DosDeviceTargets array being set up beforehand.
Arguments:
ArcPath - specifies arc path to be converted.
DriveLetter - if successful, receives letter of drive.
Return Value:
Win32 error code indicating outcome.
--*/
{ NTSTATUS Status; WCHAR drive; LPWSTR arcPath; DWORD ec;
//
// Assume failure
//
*DriveLetter = 0;
arcPath = MALLOC(((lstrlen(ArcPath)+1)*sizeof(WCHAR)) + sizeof(ArcNameDirectory)); if(!arcPath) { return(ERROR_NOT_ENOUGH_MEMORY); } lstrcpy(arcPath,ArcNameDirectory); lstrcat(arcPath,L"\\"); lstrcat(arcPath,ArcPath);
Status = QueryCanonicalName(arcPath, -1, NtName, BufferSizeInBytes); if (NT_SUCCESS(Status)) {
ec = ERROR_INVALID_DRIVE;
for(drive=L'A'; drive<=L'Z'; drive++) {
if(DosDeviceTargets[drive-L'A'] && !lstrcmpi(NtName,DosDeviceTargets[drive-L'A'])) { *DriveLetter = drive; ec = NO_ERROR; break; } }
} else { ec = RtlNtStatusToDosError(Status); }
FREE(arcPath);
return(ec); }
DWORD InitDriveNameTranslations( VOID ) { WCHAR DriveName[15]; WCHAR Drive; WCHAR Buffer[512]; NTSTATUS status;
swprintf(DriveName, L"\\DosDevices\\c:");
//
// Calculate NT names for all local hard disks C-Z.
//
for(Drive=L'A'; Drive<=L'Z'; Drive++) {
DosDeviceTargets[Drive-L'A'] = NULL;
if(MyGetDriveType(Drive) == DRIVE_FIXED) {
DriveName[12] = Drive;
status = QueryCanonicalName(DriveName, -1, Buffer, sizeof(Buffer));
if (NT_SUCCESS(status)) { DosDeviceTargets[Drive-L'A'] = DupString(Buffer); if(!DosDeviceTargets[Drive-L'A']) { return(ERROR_NOT_ENOUGH_MEMORY); } } } }
//
// Initialize old Nt Parition names to new partition name
// mapping
//
InitOldToNewNtNameTranslations();
return(NO_ERROR); }
DWORD DetermineSystemPartitions( VOID ) { LPWSTR *SyspartComponents; DWORD NumSyspartComponents; DWORD d; DWORD rc; UINT u; WCHAR drive; WCHAR DeviceNtName[512];
SyspartComponents = BootVarComponents[BootVarSystemPartition]; NumSyspartComponents = BootVarComponentCount[BootVarSystemPartition];
SystemPartitionNtNames = MALLOC ((NumSyspartComponents + 1) * sizeof (PWSTR)); if (!SystemPartitionNtNames) { return ERROR_NOT_ENOUGH_MEMORY; } ZeroMemory(SystemPartitionNtNames, (NumSyspartComponents + 1) * sizeof (PWSTR));
ZeroMemory(SystemPartitionDriveLetters,27*sizeof(WCHAR));
//
// Convert each system partition to a drive letter.
//
for(d=0; d<NumSyspartComponents; d++) { //
// check for duplicates
//
if (SystemPartitionCount > 0) { for (u = 0; u < SystemPartitionCount; u++) { if (lstrcmpi (SyspartComponents[d], SystemPartitionNtNames[u]) == 0) { break; } } if (u < SystemPartitionCount) { continue; } }
rc = ArcPathToDriveLetterAndNtName ( SyspartComponents[d], &drive, DeviceNtName, (DWORD) sizeof (DeviceNtName) ); if(rc == ERROR_NOT_ENOUGH_MEMORY) { return(ERROR_NOT_ENOUGH_MEMORY); }
if (rc == ERROR_SUCCESS) { SystemPartitionDriveLetters[SystemPartitionCount] = drive; }
SystemPartitionNtNames[SystemPartitionCount++] = DupString (DeviceNtName); }
return(NO_ERROR); }
DWORD DoInitializeArcStuff( VOID ) { DWORD ec; DWORD var; UNICODE_STRING UnicodeString; NTSTATUS Status; WCHAR Buffer[4096];
ec = InitDriveNameTranslations(); if(ec != NO_ERROR) { goto c0; }
//
// Get relevent boot vars.
//
// Enable privilege -- since we check this privilege up front
// in main() this should not fail.
//
if(!EnablePrivilege(SE_SYSTEM_ENVIRONMENT_NAME,TRUE)) { ec = ERROR_ACCESS_DENIED; goto c0; }
for(var=0; var<BootVarMax; var++) {
RtlInitUnicodeString(&UnicodeString,BootVarNames[var]);
Status = NtQuerySystemEnvironmentValue( &UnicodeString, Buffer, sizeof(Buffer) / sizeof(WCHAR), NULL );
if(NT_SUCCESS(Status)) { BootVarValues[var] = DupString(Buffer); OriginalBootVarValues[var] = DupString(Buffer); } else { //
// We may get back failure if the variable is empty.
//
BootVarValues[var] = DupString(L""); OriginalBootVarValues[var] = DupString(L""); }
if(!BootVarValues[var] || !OriginalBootVarValues[var]) { ec = ERROR_NOT_ENOUGH_MEMORY; goto c2; }
ec = GetVarComponents( BootVarValues[var], &BootVarComponents[var], &BootVarComponentCount[var] );
if(ec != NO_ERROR) { goto c2; }
//
// Track the variable with the most number of components.
//
if(BootVarComponentCount[var] > LargestComponentCount) { LargestComponentCount = BootVarComponentCount[var]; } }
//
// Get original countdown and autoload values.
// If not successful, oh well, we won't be able to restore them
// if the user cancels.
//
RtlInitUnicodeString(&UnicodeString,szCOUNTDOWN); Status = NtQuerySystemEnvironmentValue( &UnicodeString, Buffer, sizeof(Buffer) / sizeof(WCHAR), NULL ); if(NT_SUCCESS(Status)) { OriginalCountdown = DupString(Buffer); } else { OriginalCountdown = DupString(L""); }
RtlInitUnicodeString(&UnicodeString,szAUTOLOAD); Status = NtQuerySystemEnvironmentValue( &UnicodeString, Buffer, sizeof(Buffer) / sizeof(WCHAR), NULL ); if(NT_SUCCESS(Status)) { OriginalAutoload = DupString(Buffer); } else { OriginalAutoload = DupString(L"NO"); }
ec = DetermineSystemPartitions(); if(ec != NO_ERROR) { goto c2; } return(NO_ERROR);
c2: c0: return(ec); }
BOOL ArcInitializeArcStuff( IN HWND Parent ) { DWORD ec; BOOL b; HKEY key; DWORD type; DWORD size; PBYTE buffer = NULL; DWORD i;
#if defined(EFI_NVRAM_ENABLED)
//
// Try to initialize as an EFI machine. If we're on an EFI machine,
// this will succeed. Otherwise it will fail, in which case we try
// to initialize as an ARC machine.
//
ec = InitializeEfiStuff(Parent); if (!IsEfi()) #endif
{ //
// Try to initialize as an ARC machine. This is expect to
// always succeed.
//
ec = DoInitializeArcStuff(); }
switch(ec) {
case NO_ERROR:
#if defined(EFI_NVRAM_ENABLED)
//
// On an EFI machine, the rest of this code (determining system
// partitions) is not necessary.
//
if (IsEfi()) { b = TRUE; } else #endif
{ //
// Make sure there is at least one valid system partition.
//
if(!SystemPartitionCount) {
MessageBoxFromMessage( Parent, MSG_SYSTEM_PARTITION_INVALID, FALSE, AppTitleStringId, MB_OK | MB_ICONERROR | MB_TASKMODAL );
b = FALSE; } else { i = 0; //
// On ARC machines we set up a local boot directory that is
// placed in the root of the system partition.
//
//
// read the SystemPartition value from registry
//
// we must be careful in how we copy this value into buffers, since
// it comes from the registry!
//
ec = RegOpenKey (HKEY_LOCAL_MACHINE, TEXT("System\\Setup"), &key); if (ec == ERROR_SUCCESS) { ec = RegQueryValueEx (key, TEXT("SystemPartition"), NULL, &type, NULL, &size); if (ec == ERROR_SUCCESS && type == REG_SZ) { buffer = MALLOC (size); if (buffer) { ec = RegQueryValueEx (key, TEXT("SystemPartition"), NULL, &type, buffer, &size); if (ec != ERROR_SUCCESS) { FREE (buffer); buffer = NULL; } } }
RegCloseKey (key); }
#if defined(EFI_NVRAM_ENABLED)
//
// we just trust the value that comes from the regkey -- EFI
// systems only have one system partition, so it doesn't make
// sense to try to match this up against a list of potential
// system partitions.
//
SystemPartitionNtName = (PWSTR) buffer; #else
//
// look for this system partition to make sure things are OK
//
if (buffer) { while (i < SystemPartitionCount) { if (lstrcmpi (SystemPartitionNtNames[i], (PCTSTR)buffer) == 0) { SystemPartitionNtName = SystemPartitionNtNames[i]; break; } i++; } FREE (buffer); } #endif
if(!SystemPartitionNtName) {
MessageBoxFromMessage( Parent, MSG_SYSTEM_PARTITION_INVALID, FALSE, AppTitleStringId, MB_OK | MB_ICONERROR | MB_TASKMODAL );
b = FALSE;
break; }
#if !defined(EFI_NVRAM_ENABLED)
if (SystemPartitionDriveLetters[i]) { SystemPartitionDriveLetter = ForcedSystemPartition ? ForcedSystemPartition : SystemPartitionDriveLetters[i]; LocalBootDirectory[0] = SystemPartitionDriveLetter; LocalBootDirectory[1] = TEXT(':'); LocalBootDirectory[2] = TEXT('\\'); LocalBootDirectory[3] = 0; } else #endif
{
// SystemPartitionNtNtname is valid at this point thanks to
// the check above.
size = sizeof(GLOBAL_ROOT) + lstrlen(SystemPartitionNtName)*sizeof(WCHAR) + sizeof(WCHAR) + sizeof(WCHAR); SystemPartitionVolumeGuid = MALLOC (size);
if(!SystemPartitionVolumeGuid) { goto MemoryError; }
lstrcpy (SystemPartitionVolumeGuid, GLOBAL_ROOT); lstrcat (SystemPartitionVolumeGuid, SystemPartitionNtName); lstrcat (SystemPartitionVolumeGuid, L"\\");
//
// SystemPartitionVolumeGuid may contain a value from
// the registry (SystemPartitionNtName), so we have to use a
// safe string operation here.
//
if (FAILED(StringCchCopy(LocalBootDirectory, ARRAYSIZE(LocalBootDirectory), SystemPartitionVolumeGuid))) { b = FALSE; MYASSERT(FALSE); break; } }
b = TRUE; } }
break;
case ERROR_NOT_ENOUGH_MEMORY:
MemoryError:
MessageBoxFromMessage( Parent, MSG_OUT_OF_MEMORY, FALSE, AppTitleStringId, MB_OK | MB_ICONERROR | MB_TASKMODAL );
b = FALSE; break;
default: //
// Some other unknown error.
//
MessageBoxFromMessage( Parent, MSG_COULDNT_READ_NVRAM, FALSE, AppTitleStringId, MB_OK | MB_ICONERROR | MB_TASKMODAL );
b = FALSE; break; }
#if defined(EFI_NVRAM_ENABLED)
//
// make sure the system partition is on a GPT disk.
//
if (b) { HANDLE hDisk; PARTITION_INFORMATION_EX partitionEx; DWORD sizePartitionEx = 0; UNICODE_STRING uString; OBJECT_ATTRIBUTES ObjectAttributes; IO_STATUS_BLOCK IoStatus; NTSTATUS Status; PWSTR p,q;
b = FALSE;
MYASSERT( SystemPartitionVolumeGuid != NULL );
//
// SystemPartitionVolumeGuid may have a '\' at the end of it.
// delete this character or we won't open the partition properly
//
p = DupString( SystemPartitionVolumeGuid + wcslen(GLOBAL_ROOT) );
if (p) { if (wcslen(p) > 0) { if (*(p+wcslen(p)-1) == L'\\') { *(p+wcslen(p)-1) = L'\0'; } }
INIT_OBJA( &ObjectAttributes, &uString, p );
Status = NtCreateFile(&hDisk, (ACCESS_MASK)FILE_GENERIC_READ, &ObjectAttributes, &IoStatus, NULL, FILE_ATTRIBUTE_NORMAL, FILE_SHARE_READ|FILE_SHARE_WRITE, FILE_OPEN, FILE_SYNCHRONOUS_IO_NONALERT, NULL, 0 );
if (NT_SUCCESS(Status)) { Status = NtDeviceIoControlFile( hDisk, NULL, NULL, NULL, &IoStatus, IOCTL_DISK_GET_PARTITION_INFO_EX, NULL, 0, &partitionEx, sizeof(PARTITION_INFORMATION_EX) );
if (NT_SUCCESS(Status)) { if (partitionEx.PartitionStyle == PARTITION_STYLE_GPT) { b = TRUE; } } else if (Status == STATUS_INVALID_DEVICE_REQUEST) { //
// we must be running on an older build where the IOCTL
// code is different
//
Status = NtDeviceIoControlFile( hDisk, NULL, NULL, NULL, &IoStatus, CTL_CODE(IOCTL_DISK_BASE, 0x0012, METHOD_BUFFERED, FILE_READ_ACCESS), NULL, 0, &partitionEx, sizeof(PARTITION_INFORMATION_EX) );
if (NT_SUCCESS(Status)) { if (partitionEx.PartitionStyle == PARTITION_STYLE_GPT) { b = TRUE; } } }
NtClose(hDisk);
}
FREE( p ); }
if (!b) { MessageBoxFromMessage( Parent, MSG_SYSTEM_PARTITIONTYPE_INVALID, FALSE, AppTitleStringId, MB_OK | MB_ICONERROR | MB_TASKMODAL ); }
}
#endif
return(b); }
#if defined(EFI_NVRAM_ENABLED)
DWORD LocateEfiSystemPartition( OUT PWSTR SystemPartitionName ) /*++
Routine Description:
Locates the EFI system partition on a GPT disk by scanning all the available hard disks.
Arguments:
SystemPartitionName : Buffer to receive system partition name, if one is present
Return Value:
Win32 error code indicating outcome.
--*/
{ DWORD ErrorCode = ERROR_BAD_ARGUMENTS;
if (SystemPartitionName) { SYSTEM_DEVICE_INFORMATION SysDevInfo; NTSTATUS Status;
*SystemPartitionName = UNICODE_NULL;
//
// Get hold of number of hard disks on the system
//
ZeroMemory(&SysDevInfo, sizeof(SYSTEM_DEVICE_INFORMATION));
Status = NtQuerySystemInformation(SystemDeviceInformation, &SysDevInfo, sizeof(SYSTEM_DEVICE_INFORMATION), NULL);
if (NT_SUCCESS(Status)) { ULONG HardDiskCount = SysDevInfo.NumberOfDisks; ULONG CurrentDisk; ULONG BufferSize = sizeof(DRIVE_LAYOUT_INFORMATION_EX) + (sizeof(PARTITION_INFORMATION_EX) * 128); PCHAR Buffer = MALLOC(BufferSize); BOOL Found = FALSE;
if (Buffer) { //
// Go through each disk and find out its partition
// layout
//
for (CurrentDisk = 0; (!Found && (CurrentDisk < HardDiskCount)); CurrentDisk++) {
WCHAR DiskName[MAX_PATH]; HANDLE DiskHandle;
swprintf(DiskName, L"\\\\.\\PHYSICALDRIVE%d", CurrentDisk);
DiskHandle = CreateFile(DiskName, GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
if ((DiskHandle) && (DiskHandle != INVALID_HANDLE_VALUE)) { DWORD ReturnSize = 0;
ZeroMemory(Buffer, BufferSize);
if (DeviceIoControl(DiskHandle, IOCTL_DISK_GET_DRIVE_LAYOUT_EX, NULL, 0, Buffer, BufferSize, &ReturnSize, NULL)) { //
// Only search in GPT disks on IA64
//
PDRIVE_LAYOUT_INFORMATION_EX DriveLayout;
DriveLayout = (PDRIVE_LAYOUT_INFORMATION_EX)Buffer;
if (DriveLayout->PartitionStyle == PARTITION_STYLE_GPT) { ULONG PartitionIndex;
for (PartitionIndex = 0; (PartitionIndex < DriveLayout->PartitionCount); PartitionIndex++) { PPARTITION_INFORMATION_EX Partition; GUID *PartitionType;
Partition = DriveLayout->PartitionEntry + PartitionIndex; PartitionType = &(Partition->Gpt.PartitionType);
if (IsEqualGUID(PartitionType, &PARTITION_SYSTEM_GUID)) { swprintf(SystemPartitionName, L"\\Device\\Harddisk%d\\Partition%d", CurrentDisk, Partition->PartitionNumber );
Found = TRUE;
break; } } } }
CloseHandle(DiskHandle); } }
FREE(Buffer); } else { ErrorCode = ERROR_NOT_ENOUGH_MEMORY; }
if (!Found) { ErrorCode = ERROR_FILE_NOT_FOUND; } else { ErrorCode = ERROR_SUCCESS; } } }
return ErrorCode; }
DWORD InitializeEfiStuff( IN HWND Parent ) { DWORD ec; NTSTATUS status; HMODULE h; WCHAR dllName[MAX_PATH]; ULONG length; HKEY key; DWORD type; LONG i; PMY_BOOT_ENTRY bootEntry; PMY_BOOT_ENTRY previousBootEntry;
MYASSERT(!IsEfiChecked);
//
// IsEfi() uses IsEfiMachine to determine its return value. Assume that
// we're not on an EFI machine.
//
IsEfiChecked = TRUE; IsEfiMachine = FALSE;
//
// Enable the privilege that is necessary to query/set NVRAM.
//
if(!EnablePrivilege(SE_SYSTEM_ENVIRONMENT_NAME,TRUE)) { ec = GetLastError(); return ec; }
//
// Load ntdll.dll from the system directory.
//
GetSystemDirectory(dllName, MAX_PATH); ConcatenatePaths(dllName, TEXT("ntdll.dll"), MAX_PATH); h = LoadLibrary(dllName); if (h == NULL) { ec = GetLastError(); return ec; }
//
// Get the addresses of the NVRAM APIs that we need to use. If any of
// these APIs are not available, this must be a pre-EFI NVRAM build.
//
(FARPROC)AddBootEntry = GetProcAddress(h, "NtAddBootEntry"); (FARPROC)DeleteBootEntry = GetProcAddress(h, "NtDeleteBootEntry"); (FARPROC)EnumerateBootEntries = GetProcAddress(h, "NtEnumerateBootEntries"); (FARPROC)QueryBootEntryOrder = GetProcAddress(h, "NtQueryBootEntryOrder"); (FARPROC)SetBootEntryOrder = GetProcAddress(h, "NtSetBootEntryOrder"); (FARPROC)QueryBootOptions = GetProcAddress(h, "NtQueryBootOptions"); (FARPROC)SetBootOptions = GetProcAddress(h, "NtSetBootOptions");
if ((AddBootEntry == NULL) || (DeleteBootEntry == NULL) || (EnumerateBootEntries == NULL) || (QueryBootEntryOrder == NULL) || (SetBootEntryOrder == NULL) || (QueryBootOptions == NULL) || (SetBootOptions == NULL)) { return ERROR_OLD_WIN_VERSION; }
//
// Get the global system boot options. If the call fails with
// STATUS_NOT_IMPLEMENTED, this is not an EFI machine.
//
length = 0; status = QueryBootOptions(NULL, &length); if (status != STATUS_NOT_IMPLEMENTED) { IsEfiMachine = TRUE; } if (status != STATUS_BUFFER_TOO_SMALL) { if (status == STATUS_SUCCESS) { status = STATUS_UNSUCCESSFUL; } return RtlNtStatusToDosError(status); } BootOptions = MALLOC(length); OriginalBootOptions = MALLOC(length); if ((BootOptions == NULL) || (OriginalBootOptions == NULL)) { return RtlNtStatusToDosError(ERROR_NOT_ENOUGH_MEMORY); } status = QueryBootOptions(BootOptions, &length); if (status != STATUS_SUCCESS) { FREE(BootOptions); FREE(OriginalBootOptions); BootOptions = NULL; OriginalBootOptions = NULL; return RtlNtStatusToDosError(status); } memcpy(OriginalBootOptions, BootOptions, length);
//
// Get the system boot order list.
//
length = 0; status = QueryBootEntryOrder(NULL, &length); if (status != STATUS_BUFFER_TOO_SMALL) { if (status == STATUS_SUCCESS) { status = STATUS_UNSUCCESSFUL; } return RtlNtStatusToDosError(status); } OriginalBootEntryOrder = MALLOC(length * sizeof(ULONG)); if (OriginalBootEntryOrder == NULL) { return ERROR_NOT_ENOUGH_MEMORY; } status = QueryBootEntryOrder(OriginalBootEntryOrder, &length); if (status != STATUS_SUCCESS) { FREE(OriginalBootEntryOrder); OriginalBootEntryOrder = NULL; return RtlNtStatusToDosError(status); } OriginalBootEntryOrderCount = length;
//
// Get all existing boot entries.
//
length = 0; status = EnumerateBootEntries(NULL, &length); if (status != STATUS_BUFFER_TOO_SMALL) { if (status == STATUS_SUCCESS) { status = STATUS_UNSUCCESSFUL; } return RtlNtStatusToDosError(status); } BootEntries = MALLOC(length); if (BootEntries == NULL) { return ERROR_NOT_ENOUGH_MEMORY; } status = EnumerateBootEntries(BootEntries, &length); if (status != STATUS_SUCCESS) { FREE(BootEntries); BootEntries = NULL; return RtlNtStatusToDosError(status); }
//
// Initialize drive name translations, which are needed for converting
// the boot entries into their internal representations.
//
ec = InitDriveNameTranslations(); if(ec != NO_ERROR) { return ec; }
//
// Convert the boot entries into an internal representation.
//
status = ConvertBootEntries(); if (!NT_SUCCESS(status)) { return RtlNtStatusToDosError(status); }
//
// Free the enumeration buffer.
//
FREE(BootEntries); BootEntries = NULL;
//
// Boot entries are returned in an unspecified order. They are currently
// in the MyBootEntries list in the order in which they were returned.
// Sort the boot entry list based on the boot order. Do this by walking
// the boot order array backwards, reinserting the entry corresponding to
// each element of the array at the head of the list.
//
for (i = (LONG)OriginalBootEntryOrderCount - 1; i >= 0; i--) {
for (previousBootEntry = NULL, bootEntry = MyBootEntries; bootEntry != NULL; previousBootEntry = bootEntry, bootEntry = bootEntry->Next) {
if (bootEntry->NtBootEntry.Id == OriginalBootEntryOrder[i] ) {
//
// We found the boot entry with this ID. If it's not already
// at the front of the list, move it there.
//
bootEntry->Status |= MBE_STATUS_ORDERED;
if (previousBootEntry != NULL) { previousBootEntry->Next = bootEntry->Next; bootEntry->Next = MyBootEntries; MyBootEntries = bootEntry; } else { ASSERT(MyBootEntries == bootEntry); }
break; } } }
//
// Get the NT name of the system partition from the registry.
//
ec = RegOpenKey(HKEY_LOCAL_MACHINE, TEXT("System\\Setup"), &key);
if (ec == ERROR_SUCCESS) { ec = RegQueryValueEx(key, TEXT("SystemPartition"), NULL, &type, NULL, &length);
if (ec == ERROR_SUCCESS) { if (type == REG_SZ) { SystemPartitionNtName = MALLOC(length); if (SystemPartitionNtName != NULL) { ec = RegQueryValueEx( key, TEXT("SystemPartition"), NULL, &type, (PBYTE)SystemPartitionNtName, &length); if (ec != ERROR_SUCCESS) { FREE(SystemPartitionNtName); } } else { return ERROR_NOT_ENOUGH_MEMORY; } } else { return ERROR_INVALID_PARAMETER; } }
RegCloseKey (key); }
if (ec != NO_ERROR) { if (IsWinPEMode()) { WCHAR OldSysPartName[MAX_PATH] = {0}; WCHAR NewSysPartName[MAX_PATH] = {0};
ec = LocateEfiSystemPartition(OldSysPartName);
if ((ec == NO_ERROR) && OldSysPartName[0]) { NTSTATUS Status = QueryCanonicalName(OldSysPartName, -1, NewSysPartName, sizeof(NewSysPartName));
if (NT_SUCCESS(Status) && NewSysPartName[0]) { SystemPartitionNtName = DupString(NewSysPartName); } else { ec = ERROR_FILE_NOT_FOUND; } }
if ((ec == NO_ERROR) && (NewSysPartName[0] == UNICODE_NULL)) { ec = ERROR_FILE_NOT_FOUND; } }
if (ec != NO_ERROR) { return ec; } }
//
// Get the volume name for the NT name.
//
length = sizeof(GLOBAL_ROOT) + lstrlen(SystemPartitionNtName)*sizeof(WCHAR) + sizeof(WCHAR) + sizeof(WCHAR);
SystemPartitionVolumeGuid = MALLOC (length);
if(!SystemPartitionVolumeGuid) { return ERROR_NOT_ENOUGH_MEMORY; }
//
// These string operations are safe, since the buffer is preallocated
//
lstrcpy (SystemPartitionVolumeGuid, GLOBAL_ROOT); lstrcat (SystemPartitionVolumeGuid, SystemPartitionNtName); lstrcat (SystemPartitionVolumeGuid, L"\\");
//
// But strlen of SystemPartitionVolumeGuid may exceed MAX_PATH, so we
// do a safe string copy here
//
if (FAILED(StringCchCopy(LocalBootDirectory, ARRAYSIZE(LocalBootDirectory), SystemPartitionVolumeGuid))) { return ERROR_BUFFER_OVERFLOW; }
return NO_ERROR;
} // InitializeEfiStuff
#endif // defined(EFI_NVRAM_ENABLED)
/////////////////////////////////////////////////////////////////////
//
// Everything above this line is concerned with reading NV-RAM.
// Everything below this line is concerned with setting NV-RAM.
//
/////////////////////////////////////////////////////////////////////
BOOL DoSetNvRamVar( IN LPCWSTR VarName, IN LPCWSTR VarValue ) { UNICODE_STRING U1,U2;
RtlInitUnicodeString(&U1,VarName); RtlInitUnicodeString(&U2,VarValue);
return(NT_SUCCESS(NtSetSystemEnvironmentValue(&U1,&U2))); }
BOOL WriteNewBootSetVar( IN DWORD var, IN PTSTR NewPart ) { WCHAR Buffer[2048]; DWORD i;
//
// Write the new part first.
//
if (FAILED(StringCchCopy(Buffer, ARRAYSIZE(Buffer), NewPart))) { return FALSE; }
//
// Append all components that were not deleted.
//
for(i=0; i<BootVarComponentCount[var]; i++) {
if(BootVarComponents[var][i]) {
if (FAILED(StringCchCat(Buffer, ARRAYSIZE(Buffer), L";"))) { return FALSE; }
if (FAILED(StringCchCat(Buffer, ARRAYSIZE(Buffer), BootVarComponents[var][i]))) { return FALSE; } } }
//
// Remember new value for this var.
//
if(BootVarValues[var]) { FREE(BootVarValues[var]); }
BootVarValues[var] = DupString(Buffer);
if (!BootVarValues[var]) return FALSE;
//
// Write the var into nvram and return.
//
return(DoSetNvRamVar(BootVarNames[var],BootVarValues[var])); }
BOOL WriteBootSet( VOID ) { DWORD set; DWORD var; LPWSTR SystemPartition; WCHAR Buffer[2048]; LPWSTR LocalSourceArc; LPWSTR OsLoader; WCHAR LoadId[128]; BOOL b;
CleanUpNvRam = TRUE;
//
// Find and remove any remnants of previously attempted
// winnt32 runs. Such runs are identified by 'winnt32'
// in their osloadoptions.
//
#if defined(EFI_NVRAM_ENABLED)
if (IsEfi()) {
NTSTATUS status; PMY_BOOT_ENTRY bootEntry; PWSTR NtPath;
//
// EFI machine. Walk the boot entry list.
//
for (bootEntry = MyBootEntries; bootEntry != NULL; bootEntry = bootEntry->Next) {
if (IS_BOOT_ENTRY_WINDOWS(bootEntry)) {
if (!lstrcmpi(bootEntry->OsLoadOptions, L"WINNT32")) {
//
// Delete this boot entry. Note that we don't update the
// boot entry order list at this point. CreateBootEntry()
// will do that.
//
status = DeleteBootEntry(bootEntry->NtBootEntry.Id);
bootEntry->Status |= MBE_STATUS_DELETED; } } }
//
// Now create a new boot entry for textmode setup.
//
MYASSERT(LocalSourceDrive); NtPath = DosDeviceTargets[(WCHAR)CharUpper((PWCHAR)LocalSourceDrive)-L'A'];
LoadString(hInst,IDS_RISCBootString,LoadId,sizeof(LoadId)/sizeof(TCHAR));
b = CreateBootEntry( SystemPartitionNtName, L"\\" SETUPLDR_FILENAME, NtPath, LocalSourceWithPlatform + 2, L"WINNT32", LoadId );
if (b) {
//
// Set up for automatic startup, 10 second countdown. We don't
// care if this fails.
//
// Set the boot entry we added to be booted automatically on
// the next boot, without waiting for a timeout at the boot menu.
//
// NB: CreateBootEntry() sets BootOptions->NextBootEntryId.
//
BootOptions->Timeout = 10; status = SetBootOptions( BootOptions, BOOT_OPTIONS_FIELD_TIMEOUT | BOOT_OPTIONS_FIELD_NEXT_BOOT_ENTRY_ID ); }
return b; }
#endif // defined(EFI_NVRAM_ENABLED)
//
// We get here if we're NOT on an EFI machine.
//
// Find and remove any remnants of previously attempted
// winnt32 runs. Such runs are identified by 'winnt32'
// in their osloadoptions.
//
for(set=0; set<min(LargestComponentCount,BootVarComponentCount[BootVarOsLoadOptions]); set++) {
//
// See if the os load options indicate that this is a winnt32 set.
//
if(!lstrcmpi(BootVarComponents[BootVarOsLoadOptions][set],L"WINNT32")) {
//
// Delete this boot set.
//
for(var=0; var<BootVarMax; var++) {
if(set < BootVarComponentCount[var]) {
FREE(BootVarComponents[var][set]); BootVarComponents[var][set] = NULL; } } } }
//
// Now we want to write out each variable with the appropriate
// part of the new boot set added to the front.
//
if (SystemPartitionDriveLetter) { if(DriveLetterToArcPath(SystemPartitionDriveLetter,&SystemPartition) != NO_ERROR) { return(FALSE); } } else { if(NtNameToArcPath (SystemPartitionNtName, &SystemPartition) != NO_ERROR) { return(FALSE); } } MYASSERT (LocalSourceDrive); if(DriveLetterToArcPath(LocalSourceDrive,&LocalSourceArc) != NO_ERROR) { FREE(SystemPartition); return(FALSE); }
LoadString(hInst,IDS_RISCBootString,LoadId,sizeof(LoadId)/sizeof(TCHAR));
if (FAILED(StringCchCopy(Buffer, ARRAYSIZE(Buffer), SystemPartition))) { FREE(SystemPartition); FREE(LocalSourceArc); return (FALSE); }
if (FAILED(StringCchCat(Buffer, ARRAYSIZE(Buffer), L"\\" SETUPLDR_FILENAME))) { FREE(SystemPartition); FREE(LocalSourceArc); return (FALSE); }
OsLoader = DupString(Buffer);
if (!OsLoader) { FREE(SystemPartition); FREE(LocalSourceArc); return (FALSE); }
//
// System partition: use the selected system partition as the
// new system partition component.
//
if(WriteNewBootSetVar(BootVarSystemPartition,SystemPartition)
//
// Os Loader: use the system partition + setupldr as the
// new os loader component.
//
&& WriteNewBootSetVar(BootVarOsLoader,OsLoader)
//
// Os Load Partition: use the local source drive as the
// new os load partition component.
//
&& WriteNewBootSetVar(BootVarOsLoadPartition,LocalSourceArc)
//
// Os Load Filename: use the platform-specific local source directory
// as the new os load filename component (do not include the drive letter).
//
&& WriteNewBootSetVar(BootVarOsLoadFilename,LocalSourceWithPlatform+2)
//
// Os Load Options: use WINNT32 as the new os load options component.
//
&& WriteNewBootSetVar(BootVarOsLoadOptions,L"WINNT32")
//
// Load Identifier: use a string we get from the resources as the
// new load identifier component.
//
&& WriteNewBootSetVar(BootVarLoadIdentifier,LoadId)) { //
// Set up for automatic startup, 10 second countdown.
// Note the order so that if setting countdown fails we don't
// set of for autoload. Also note that we don't really care
// if this fails.
//
if(DoSetNvRamVar(szCOUNTDOWN,L"10")) { DoSetNvRamVar(szAUTOLOAD,L"YES"); }
b = TRUE;
} else { //
// Setting nv-ram failed. Code in cleanup.c will come along and
// restore to original state later.
//
b = FALSE; }
FREE(SystemPartition); FREE(LocalSourceArc); FREE(OsLoader);
return(b); }
BOOL SetUpNvRam( IN HWND ParentWindow ) { if(!WriteBootSet()) {
MessageBoxFromMessage( ParentWindow, MSG_COULDNT_WRITE_NVRAM, FALSE, AppTitleStringId, MB_OK | MB_ICONERROR | MB_TASKMODAL );
return(FALSE); }
return(TRUE); }
BOOL RestoreNvRam( VOID ) { UINT var; BOOL b;
b = TRUE;
if(CleanUpNvRam) {
#if defined(EFI_NVRAM_ENABLED)
if (IsEfi()) {
NTSTATUS status; PMY_BOOT_ENTRY bootEntry;
//
// EFI machine. Walk the boot entry list. For any boot entry that
// was added, delete it.
//
for (bootEntry = MyBootEntries; bootEntry != NULL; bootEntry = bootEntry->Next) { if (IS_BOOT_ENTRY_COMMITTED(bootEntry)) { MYASSERT(IS_BOOT_ENTRY_NEW(bootEntry)); status = DeleteBootEntry(bootEntry->NtBootEntry.Id); if (!NT_SUCCESS(status)) { b = FALSE; } } }
//
// Restore the original boot order list and the original timeout.
//
status = SetBootEntryOrder(OriginalBootEntryOrder, OriginalBootEntryOrderCount); if (!NT_SUCCESS(status)) { b = FALSE; }
status = SetBootOptions(OriginalBootOptions, BOOT_OPTIONS_FIELD_TIMEOUT); if (!NT_SUCCESS(status)) { b = FALSE; } }
} else {
#endif // defined(EFI_NVRAM_ENABLED)
for(var=0; var<BootVarMax; var++) { if(!DoSetNvRamVar(BootVarNames[var],OriginalBootVarValues[var])) { b = FALSE; } }
if(OriginalAutoload) { if(!DoSetNvRamVar(szAUTOLOAD,OriginalAutoload)) { b = FALSE; } } if(OriginalCountdown) { if(!DoSetNvRamVar(szCOUNTDOWN,OriginalCountdown)) { b = FALSE; } } }
return(b); }
VOID MigrateBootVarData( VOID ) /*++
Routine Description:
This routine retreives any boot data we want to migrate into a global variable so that it can be written into winnt.sif.
Currently we only retreive the countdown
Arguments:
None
Return Value:
None. updates the Timeout global variable
--*/ { UNICODE_STRING UnicodeString; NTSTATUS Status; WCHAR Buffer[4096];
MYASSERT(IsArc());
//
// If this is an EFI machine, use the cached BootOptions to get the timeout.
// (See IsEfi().) Otherwise, use the old version of the system service to
// query the "COUNTDOWN" variable.
//
#if defined(EFI_NVRAM_ENABLED)
if (IsEfi()) {
MYASSERT(BootOptions != NULL);
swprintf( Timeout, L"%d", BootOptions->Timeout );
} else
#endif // defined(EFI_NVRAM_ENABLED)
{ RtlInitUnicodeString(&UnicodeString,szCOUNTDOWN); Status = NtQuerySystemEnvironmentValue( &UnicodeString, Buffer, sizeof(Buffer) / sizeof(WCHAR), NULL ); if(NT_SUCCESS(Status)) {
//
// Global Timeout buffer is only 32 TCHARs, so use safe string copy!
//
StringCchCopy(Timeout, ARRAYSIZE(Timeout), Buffer); } }
}
#if defined(_X86_)
BOOL IsArc( VOID )
/*++
Routine Description:
Run time check to determine if this is an Arc system. We attempt to read an Arc variable using the Hal. This will fail for Bios based systems.
Arguments:
None
Return Value:
True = This is an Arc system.
--*/
{ UNICODE_STRING UnicodeString; NTSTATUS Status; WCHAR Buffer[4096];
//
// If we've already done the check once, don't bother doing it again.
//
if (IsArcChecked) { return IsArcMachine; }
IsArcChecked = TRUE; IsArcMachine = FALSE;
if(!EnablePrivilege(SE_SYSTEM_ENVIRONMENT_NAME,TRUE)) { return FALSE; // need better error handling?
}
//
// Get the env var into the temp buffer.
//
RtlInitUnicodeString(&UnicodeString,BootVarNames[BootVarOsLoader]);
Status = NtQuerySystemEnvironmentValue( &UnicodeString, Buffer, sizeof(Buffer)/sizeof(WCHAR), NULL );
if (NT_SUCCESS(Status)) { IsArcMachine = TRUE; }
return IsArcMachine; }
#endif // defined(_X86_)
#if defined(EFI_NVRAM_ENABLED)
BOOL IsEfi( VOID )
/*++
Routine Description:
Run time check to determine if this is an EFI system.
Arguments:
None
Return Value:
True = This is an EFI system.
--*/
{ //
// InitializeEfiStuff() must be called first to do the actual check.
//
MYASSERT(IsEfiChecked);
return IsEfiMachine;
} // IsEfi
NTSTATUS ConvertBootEntries( VOID )
/*++
Routine Description:
Convert boot entries read from EFI NVRAM into our internal format.
Arguments:
None.
Return Value:
NTSTATUS - Not STATUS_SUCCESS if an unexpected error occurred.
--*/
{ PBOOT_ENTRY_LIST bootEntryList; PBOOT_ENTRY bootEntry; PBOOT_ENTRY bootEntryCopy; PMY_BOOT_ENTRY myBootEntry; PMY_BOOT_ENTRY previousEntry; PWINDOWS_OS_OPTIONS osOptions; ULONG length;
bootEntryList = BootEntries; previousEntry = NULL;
while (TRUE) {
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;
myBootEntry = MALLOC(length); if (myBootEntry == NULL) { return STATUS_INSUFFICIENT_RESOURCES; }
RtlZeroMemory(myBootEntry, length);
//
// Link the new entry into the list.
//
if (previousEntry != NULL) { previousEntry->Next = myBootEntry; } else { MyBootEntries = myBootEntry; } previousEntry = myBootEntry;
//
// Copy the NT BOOT_ENTRY into the allocated buffer.
//
bootEntryCopy = &myBootEntry->NtBootEntry;
//
// work around till bootentry has the correct length specified
//
__try { memcpy(bootEntryCopy, bootEntry, bootEntry->Length); } __except(EXCEPTION_EXECUTE_HANDLER) { if (bootEntry->Length > sizeof(ULONG)) { bootEntry->Length -= sizeof(ULONG); memcpy(bootEntryCopy, bootEntry, bootEntry->Length); } else { //
// Lets atleast AV rather than having invalid
// in memory data structures
//
memcpy(bootEntryCopy, bootEntry, bootEntry->Length); } }
//
// Fill in the base part of the structure.
//
myBootEntry->Next = NULL; myBootEntry->AllocationEnd = (PUCHAR)myBootEntry + length - 1; myBootEntry->FriendlyName = ADD_OFFSET(bootEntryCopy, FriendlyNameOffset); myBootEntry->FriendlyNameLength = (wcslen(myBootEntry->FriendlyName) + 1) * sizeof(WCHAR); myBootEntry->BootFilePath = 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 (!IS_BOOT_ENTRY_WINDOWS(myBootEntry)) {
//
// The original implementation of NtEnumerateBootEntries() didn't
// set BOOT_ENTRY_ATTRIBUTE_WINDOWS, so we need to check for that
// here.
//
if ((bootEntryCopy->OsOptionsLength >= FIELD_OFFSET(WINDOWS_OS_OPTIONS, OsLoadOptions)) && (strcmp(osOptions->Signature, WINDOWS_OS_OPTIONS_SIGNATURE) == 0)) { myBootEntry->NtBootEntry.Attributes |= BOOT_ENTRY_ATTRIBUTE_WINDOWS; } }
if (IS_BOOT_ENTRY_WINDOWS(myBootEntry)) {
myBootEntry->OsLoadOptions = osOptions->OsLoadOptions; myBootEntry->OsLoadOptionsLength = (wcslen(myBootEntry->OsLoadOptions) + 1) * sizeof(WCHAR); myBootEntry->OsFilePath = ADD_OFFSET(osOptions, OsLoadPathOffset);
} else {
//
// It's not an NT entry. Check to see if it represents a removable
// media device. We want to know this so that we don't put our
// boot entry ahead of the floppy or the CD, if they're already
// at the front of the list. A boot entry represents a
}
//
// Move to the next entry in the enumeration list, if any.
//
if (bootEntryList->NextEntryOffset == 0) { break; }
bootEntryList = ADD_OFFSET(bootEntryList, NextEntryOffset); }
return STATUS_SUCCESS;
} // ConvertBootEntries
BOOL CreateBootEntry( PWSTR BootFileDevice, PWSTR BootFilePath, PWSTR OsLoadDevice, PWSTR OsLoadPath, PWSTR OsLoadOptions, PWSTR FriendlyName )
/*++
Routine Description:
Create an internal-format boot entry.
Arguments:
BootFileDevice - The NT name of the device on which the OS loader resides.
BootFilePath - The volume-relative path to the OS loader. Must start with a backslash.
OsLoadDevice - The NT name ofthe device on which the OS resides.
OsLoadPath - The volume-relative path to the OS root directory (\WINDOWS). Must start with a backslash.
OsLoadOptions - Boot options for the OS. Can be an empty string.
FriendlyName - The user-visible name for the boot entry. (This is ARC's LOADIDENTIFIER.)
Return Value:
BOOLEAN - FALSE if an unexpected error occurred.
--*/
{ NTSTATUS status; ULONG requiredLength; ULONG osOptionsOffset; ULONG osLoadOptionsLength; ULONG osLoadPathOffset; ULONG osLoadPathLength; ULONG osOptionsLength; ULONG friendlyNameOffset; ULONG friendlyNameLength; ULONG bootPathOffset; ULONG bootPathLength; PMY_BOOT_ENTRY myBootEntry; PMY_BOOT_ENTRY previousBootEntry; PMY_BOOT_ENTRY nextBootEntry; PBOOT_ENTRY ntBootEntry; PWINDOWS_OS_OPTIONS osOptions; PFILE_PATH osLoadPath; PWSTR friendlyName; PFILE_PATH bootPath; PWSTR p; PULONG order; ULONG count; ULONG savedAttributes;
//
// 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; requiredLength += FIELD_OFFSET(WINDOWS_OS_OPTIONS, OsLoadOptions); osLoadOptionsLength = (wcslen(OsLoadOptions) + 1) * sizeof(WCHAR); requiredLength += osLoadOptionsLength;
//
// Round up to a ULONG boundary for the OS FILE_PATH in the
// WINDOWS_OS_OPTIONS. Save offset to OS FILE_PATH. Add in base part
// of FILE_PATH. Add in length in bytes of OS device NT name and OS
// directory. Calculate total length of OS FILE_PATH and of
// WINDOWS_OS_OPTIONS.
//
requiredLength = ALIGN_UP(requiredLength, ULONG); osLoadPathOffset = requiredLength; requiredLength += FIELD_OFFSET(FILE_PATH, FilePath); requiredLength += (wcslen(OsLoadDevice) + 1 + wcslen(OsLoadPath) + 1) * sizeof(WCHAR); osLoadPathLength = requiredLength - osLoadPathOffset; 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 = (wcslen(FriendlyName) + 1) * sizeof(WCHAR); requiredLength += friendlyNameLength;
//
// Round up to a ULONG boundary for the boot FILE_PATH in the BOOT_ENTRY.
// Save offset to boot FILE_PATH. Add in base part of FILE_PATH. Add in
// length in bytes of boot device NT name and boot file. Calculate total
// length of boot FILE_PATH.
//
requiredLength = ALIGN_UP(requiredLength, ULONG); bootPathOffset = requiredLength; requiredLength += FIELD_OFFSET(FILE_PATH, FilePath); requiredLength += (wcslen(BootFileDevice) + 1 + wcslen(BootFilePath) + 1) * sizeof(WCHAR); bootPathLength = requiredLength - bootPathOffset;
//
// Allocate memory for the boot entry.
//
myBootEntry = MALLOC(requiredLength); if (myBootEntry == NULL) { return FALSE; }
RtlZeroMemory(myBootEntry, requiredLength);
//
// Calculate addresses of various substructures using the saved offsets.
//
ntBootEntry = &myBootEntry->NtBootEntry; osOptions = (PWINDOWS_OS_OPTIONS)ntBootEntry->OsOptions; osLoadPath = (PFILE_PATH)((PUCHAR)myBootEntry + osLoadPathOffset); friendlyName = (PWSTR)((PUCHAR)myBootEntry + friendlyNameOffset); bootPath = (PFILE_PATH)((PUCHAR)myBootEntry + bootPathOffset);
//
// Fill in the internal-format structure.
//
myBootEntry->AllocationEnd = (PUCHAR)myBootEntry + requiredLength; myBootEntry->Status = MBE_STATUS_NEW | MBE_STATUS_ORDERED; myBootEntry->FriendlyName = friendlyName; myBootEntry->FriendlyNameLength = friendlyNameLength; myBootEntry->OsLoadOptions = osOptions->OsLoadOptions; myBootEntry->OsLoadOptionsLength = osLoadOptionsLength; myBootEntry->BootFilePath = bootPath; myBootEntry->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 = BOOT_ENTRY_ATTRIBUTE_ACTIVE | BOOT_ENTRY_ATTRIBUTE_WINDOWS; ntBootEntry->FriendlyNameOffset = (ULONG)((PUCHAR)friendlyName - (PUCHAR)ntBootEntry); ntBootEntry->BootFilePathOffset = (ULONG)((PUCHAR)bootPath - (PUCHAR)ntBootEntry); ntBootEntry->OsOptionsLength = osOptionsLength;
//
// Fill in the base part of the WINDOWS_OS_OPTIONS, including the
// OsLoadOptions.
//
strcpy(osOptions->Signature, WINDOWS_OS_OPTIONS_SIGNATURE); osOptions->Version = WINDOWS_OS_OPTIONS_VERSION; osOptions->Length = osOptionsLength; osOptions->OsLoadPathOffset = (ULONG)((PUCHAR)osLoadPath - (PUCHAR)osOptions); wcscpy(osOptions->OsLoadOptions, OsLoadOptions);
//
// Fill in the OS FILE_PATH.
//
osLoadPath->Version = FILE_PATH_VERSION; osLoadPath->Length = osLoadPathLength; osLoadPath->Type = FILE_PATH_TYPE_NT; p = (PWSTR)osLoadPath->FilePath; wcscpy(p, OsLoadDevice); p += wcslen(p) + 1; wcscpy(p, OsLoadPath);
//
// Copy the friendly name.
//
wcscpy(friendlyName, FriendlyName);
//
// Fill in the boot FILE_PATH.
//
bootPath->Version = FILE_PATH_VERSION; bootPath->Length = bootPathLength; bootPath->Type = FILE_PATH_TYPE_NT; p = (PWSTR)bootPath->FilePath; wcscpy(p, BootFileDevice); p += wcslen(p) + 1; wcscpy(p, BootFilePath);
//
// Add the new boot entry.
//
// NB: The original implementation of NtAddBootEntry didn't like it
// when attribute bits other than _ACTIVE and _DEFAULT were set, so
// we need to mask the other bits off here.
//
savedAttributes = ntBootEntry->Attributes; ntBootEntry->Attributes &= (BOOT_ENTRY_ATTRIBUTE_DEFAULT | BOOT_ENTRY_ATTRIBUTE_ACTIVE); status = AddBootEntry(ntBootEntry, &ntBootEntry->Id); ntBootEntry->Attributes = savedAttributes; if (!NT_SUCCESS(status)) { FREE(myBootEntry); return FALSE; } myBootEntry->Status |= MBE_STATUS_COMMITTED;
//
// Remember the ID of the new boot entry as the entry to be booted
// immediately on the next boot.
//
BootOptions->NextBootEntryId = ntBootEntry->Id;
//
// Link the new boot entry into the list, after any removable media
// entries that are at the front of the list.
//
previousBootEntry = NULL; nextBootEntry = MyBootEntries; while ((nextBootEntry != NULL) && IS_BOOT_ENTRY_REMOVABLE_MEDIA(nextBootEntry)) { previousBootEntry = nextBootEntry; nextBootEntry = nextBootEntry->Next; } myBootEntry->Next = nextBootEntry; if (previousBootEntry == NULL) { MyBootEntries = myBootEntry; } else { previousBootEntry->Next = myBootEntry; }
//
// Build the new boot order list. Insert all boot entries with
// MBE_STATUS_ORDERED into the list. (Don't insert deleted entries.)
//
count = 0; nextBootEntry = MyBootEntries; while (nextBootEntry != NULL) { if (IS_BOOT_ENTRY_ORDERED(nextBootEntry) && !IS_BOOT_ENTRY_DELETED(nextBootEntry)) { count++; } nextBootEntry = nextBootEntry->Next; } order = MALLOC(count * sizeof(ULONG)); if (order == NULL) { return FALSE; } count = 0; nextBootEntry = MyBootEntries; while (nextBootEntry != NULL) { if (IS_BOOT_ENTRY_ORDERED(nextBootEntry) && !IS_BOOT_ENTRY_DELETED(nextBootEntry)) { order[count++] = nextBootEntry->NtBootEntry.Id; } nextBootEntry = nextBootEntry->Next; }
//
// Write the new boot entry order list.
//
status = SetBootEntryOrder(order, count); FREE(order); if (!NT_SUCCESS(status)) { return FALSE; }
return TRUE;
} // CreateBootEntry
#endif // defined(EFI_NVRAM_ENABLED)
#endif // UNICODE
|