Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

3048 lines
84 KiB

/*++
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