/** @file
OpenCore driver.
Copyright (c) 2019, vit9696. All rights reserved.
This program and the accompanying materials
are licensed and made available under the terms and conditions of the BSD License
which accompanies this distribution. The full text of the license may be found at
http://opensource.org/licenses/bsd-license.php
THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
**/
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
STATIC
OC_GLOBAL_CONFIG
mOpenCoreConfiguration;
STATIC
OC_STORAGE_CONTEXT
mOpenCoreStorage;
STATIC
OC_CPU_INFO
mOpenCoreCpuInfo;
STATIC
OC_RSA_PUBLIC_KEY *
mOpenCoreVaultKey;
STATIC
OC_PRIVILEGE_CONTEXT
mOpenCorePrivilege;
STATIC
EFI_HANDLE
mStorageHandle;
STATIC
EFI_DEVICE_PATH_PROTOCOL *
mStoragePath;
STATIC
CHAR16 *
mStorageRoot;
STATIC
EFI_STATUS
EFIAPI
OcStartImage (
IN OC_BOOT_ENTRY *Chosen,
IN EFI_HANDLE ImageHandle,
OUT UINTN *ExitDataSize,
OUT CHAR16 **ExitData OPTIONAL,
IN BOOLEAN LaunchInText
)
{
EFI_STATUS Status;
EFI_CONSOLE_CONTROL_SCREEN_MODE OldMode;
OldMode = OcConsoleControlSetMode (
LaunchInText ? EfiConsoleControlScreenText : EfiConsoleControlScreenGraphics
);
Status = gBS->StartImage (
ImageHandle,
ExitDataSize,
ExitData
);
if (EFI_ERROR (Status)) {
DEBUG ((DEBUG_WARN, "OC: Boot failed - %r\n", Status));
}
OcConsoleControlSetMode (OldMode);
return Status;
}
STATIC
VOID
OcMain (
IN OC_STORAGE_CONTEXT *Storage,
IN EFI_DEVICE_PATH_PROTOCOL *LoadPath OPTIONAL
)
{
EFI_STATUS Status;
OC_PRIVILEGE_CONTEXT *Privilege;
DEBUG ((DEBUG_INFO, "OC: OcMiscEarlyInit...\n"));
Status = OcMiscEarlyInit (
Storage,
&mOpenCoreConfiguration,
mOpenCoreVaultKey
);
if (EFI_ERROR (Status)) {
return;
}
OcCpuScanProcessor (&mOpenCoreCpuInfo);
DEBUG ((DEBUG_INFO, "OC: OcLoadNvramSupport...\n"));
OcLoadNvramSupport (Storage, &mOpenCoreConfiguration);
DEBUG ((DEBUG_INFO, "OC: OcMiscMiddleInit...\n"));
OcMiscMiddleInit (Storage, &mOpenCoreConfiguration, mStorageRoot, LoadPath, mStorageHandle);
DEBUG ((DEBUG_INFO, "OC: OcLoadUefiSupport...\n"));
OcLoadUefiSupport (Storage, &mOpenCoreConfiguration, &mOpenCoreCpuInfo);
DEBUG ((DEBUG_INFO, "OC: OcLoadAcpiSupport...\n"));
OcLoadAcpiSupport (&mOpenCoreStorage, &mOpenCoreConfiguration);
DEBUG ((DEBUG_INFO, "OC: OcLoadPlatformSupport...\n"));
OcLoadPlatformSupport (&mOpenCoreConfiguration, &mOpenCoreCpuInfo);
DEBUG ((DEBUG_INFO, "OC: OcLoadDevPropsSupport...\n"));
OcLoadDevPropsSupport (&mOpenCoreConfiguration);
DEBUG ((DEBUG_INFO, "OC: OcMiscLateInit...\n"));
OcMiscLateInit (Storage, &mOpenCoreConfiguration);
DEBUG ((DEBUG_INFO, "OC: OcLoadKernelSupport...\n"));
OcLoadKernelSupport (&mOpenCoreStorage, &mOpenCoreConfiguration, &mOpenCoreCpuInfo);
if (mOpenCoreConfiguration.Misc.Security.EnablePassword) {
mOpenCorePrivilege.CurrentLevel = OcPrivilegeUnauthorized;
mOpenCorePrivilege.Hash = mOpenCoreConfiguration.Misc.Security.PasswordHash;
mOpenCorePrivilege.Salt = OC_BLOB_GET (&mOpenCoreConfiguration.Misc.Security.PasswordSalt);
mOpenCorePrivilege.SaltSize = mOpenCoreConfiguration.Misc.Security.PasswordSalt.Size;
Privilege = &mOpenCorePrivilege;
} else {
Privilege = NULL;
}
DEBUG ((DEBUG_INFO, "OC: All green, starting boot management...\n"));
OcMiscBoot (
&mOpenCoreStorage,
&mOpenCoreConfiguration,
Privilege,
OcStartImage,
mOpenCoreConfiguration.Uefi.Quirks.RequestBootVarRouting,
mStorageHandle
);
}
STATIC
EFI_STATUS
EFIAPI
OcBootstrapRerun (
IN OC_BOOTSTRAP_PROTOCOL *This,
IN EFI_SIMPLE_FILE_SYSTEM_PROTOCOL *FileSystem,
IN EFI_DEVICE_PATH_PROTOCOL *LoadPath OPTIONAL
)
{
EFI_STATUS Status;
EFI_DEVICE_PATH_PROTOCOL *RemainingPath;
UINTN StoragePathSize;
DEBUG ((DEBUG_INFO, "OC: ReRun executed!\n"));
++This->NestedCount;
if (This->NestedCount == 1) {
mOpenCoreVaultKey = OcGetVaultKey (This);
//
// Calculate root path (never freed).
//
RemainingPath = NULL;
if (LoadPath != NULL) {
ASSERT (mStorageRoot == NULL);
mStorageRoot = OcCopyDevicePathFullName (LoadPath, &RemainingPath);
//
// Skipping this or later failing to call UnicodeGetParentDirectory means
// we got valid path to the root of the partition. This happens when
// OpenCore.efi was loaded from e.g. firmware and then bootstrapped
// on a different partition.
//
if (mStorageRoot != NULL) {
if (UnicodeGetParentDirectory (mStorageRoot)) {
//
// This means we got valid path to ourselves.
//
DEBUG ((DEBUG_INFO, "OC: Got launch root path %s\n", mStorageRoot));
} else {
FreePool (mStorageRoot);
mStorageRoot = NULL;
}
}
}
if (mStorageRoot == NULL) {
mStorageRoot = OPEN_CORE_ROOT_PATH;
RemainingPath = NULL;
DEBUG ((DEBUG_INFO, "OC: Got default root path %s\n", mStorageRoot));
}
//
// Calculate storage path.
//
if (RemainingPath != NULL) {
StoragePathSize = (UINTN) RemainingPath - (UINTN) LoadPath;
mStoragePath = AllocatePool (StoragePathSize + END_DEVICE_PATH_LENGTH);
if (mStoragePath != NULL) {
CopyMem (mStoragePath, LoadPath, StoragePathSize);
SetDevicePathEndNode ((UINT8 *) mStoragePath + StoragePathSize);
}
} else {
mStoragePath = NULL;
}
RemainingPath = LoadPath;
gBS->LocateDevicePath (
&gEfiSimpleFileSystemProtocolGuid,
&RemainingPath,
&mStorageHandle
);
Status = OcStorageInitFromFs (
&mOpenCoreStorage,
FileSystem,
mStorageHandle,
mStoragePath,
mStorageRoot,
mOpenCoreVaultKey
);
if (!EFI_ERROR (Status)) {
OcMain (&mOpenCoreStorage, LoadPath);
OcStorageFree (&mOpenCoreStorage);
} else {
DEBUG ((DEBUG_ERROR, "OC: Failed to open root FS - %r!\n", Status));
if (Status == EFI_SECURITY_VIOLATION) {
CpuDeadLoop (); ///< Should not return.
}
}
} else {
DEBUG ((DEBUG_WARN, "OC: Nested ReRun is not supported\n"));
Status = EFI_ALREADY_STARTED;
}
--This->NestedCount;
return Status;
}
STATIC
EFI_HANDLE
EFIAPI
OcGetLoadHandle (
IN OC_BOOTSTRAP_PROTOCOL *This
)
{
return mStorageHandle;
}
STATIC
OC_BOOTSTRAP_PROTOCOL
mOpenCoreBootStrap = {
.Revision = OC_BOOTSTRAP_PROTOCOL_REVISION,
.NestedCount = 0,
.VaultKey = NULL,
.ReRun = OcBootstrapRerun,
.GetLoadHandle = OcGetLoadHandle,
};
EFI_STATUS
EFIAPI
UefiMain (
IN EFI_HANDLE ImageHandle,
IN EFI_SYSTEM_TABLE *SystemTable
)
{
EFI_STATUS Status;
EFI_LOADED_IMAGE_PROTOCOL *LoadedImage;
EFI_SIMPLE_FILE_SYSTEM_PROTOCOL *FileSystem;
EFI_HANDLE BootstrapHandle;
OC_BOOTSTRAP_PROTOCOL *Bootstrap;
EFI_DEVICE_PATH_PROTOCOL *AbsPath;
DEBUG ((DEBUG_INFO, "OC: Starting OpenCore...\n"));
//
// We have just started by bootstrap or manually at EFI/OC/OpenCore.efi.
// When bootstrap runs us, we only install the protocol.
// Otherwise we do self start.
//
Bootstrap = NULL;
Status = gBS->LocateProtocol (
&gOcBootstrapProtocolGuid,
NULL,
(VOID **) &Bootstrap
);
if (!EFI_ERROR (Status)) {
DEBUG ((DEBUG_INFO, "OC: Found previous image, aborting\n"));
return EFI_ALREADY_STARTED;
}
LoadedImage = NULL;
Status = gBS->HandleProtocol (
ImageHandle,
&gEfiLoadedImageProtocolGuid,
(VOID **) &LoadedImage
);
if (EFI_ERROR (Status)) {
DEBUG ((DEBUG_ERROR, "OC: Failed to locate loaded image - %r\n", Status));
return EFI_NOT_FOUND;
}
BootstrapHandle = NULL;
Status = gBS->InstallMultipleProtocolInterfaces (
&BootstrapHandle,
&gOcBootstrapProtocolGuid,
&mOpenCoreBootStrap,
NULL
);
if (EFI_ERROR (Status)) {
DEBUG ((DEBUG_ERROR, "OC: Failed to install bootstrap protocol - %r\n", Status));
return Status;
}
DebugPrintDevicePath (DEBUG_INFO, "OC: Booter path", LoadedImage->FilePath);
if (LoadedImage->FilePath == NULL) {
DEBUG ((DEBUG_INFO, "OC: Booted from bootstrap\n"));
return EFI_SUCCESS;
}
//
// Obtain the file system device path
//
FileSystem = LocateFileSystem (
LoadedImage->DeviceHandle,
LoadedImage->FilePath
);
AbsPath = AbsoluteDevicePath (LoadedImage->DeviceHandle, LoadedImage->FilePath);
//
// Return success in either case to let rerun work afterwards.
//
if (FileSystem != NULL) {
mOpenCoreBootStrap.ReRun (&mOpenCoreBootStrap, FileSystem, AbsPath);
DEBUG ((DEBUG_ERROR, "OC: Failed to boot\n"));
} else {
DEBUG ((DEBUG_ERROR, "OC: Failed to locate file system\n"));
}
if (AbsPath != NULL) {
FreePool (AbsPath);
}
return EFI_SUCCESS;
}