mirror of
https://github.com/CloverHackyColor/CloverBootloader.git
synced 2025-01-27 21:51:31 +01:00
cd23181296
Signed-off-by: SergeySlice <sergey.slice@gmail.com>
272 lines
6.9 KiB
C
272 lines
6.9 KiB
C
/** @file
|
|
Copyright (C) 2019, vit9696. All rights reserved.
|
|
|
|
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 <Guid/AppleVariable.h>
|
|
|
|
#include <IndustryStandard/AppleHibernate.h>
|
|
|
|
#include <Library/MemoryAllocationLib.h>
|
|
#include <Library/BaseMemoryLib.h>
|
|
#include <Library/DevicePathLib.h>
|
|
#include <Library/OcBootManagementLib.h>
|
|
#include <Library/OcCryptoLib.h>
|
|
#include <Library/OcDevicePathLib.h>
|
|
#include <Library/OcDebugLogLib.h>
|
|
#include <Library/OcRtcLib.h>
|
|
#include <Library/UefiLib.h>
|
|
#include <Library/UefiRuntimeServicesTableLib.h>
|
|
|
|
EFI_STATUS
|
|
OcActivateHibernateWake (
|
|
IN UINT32 HibernateMask
|
|
)
|
|
{
|
|
EFI_STATUS Status;
|
|
UINTN Size;
|
|
UINT32 Attributes;
|
|
VOID *Value;
|
|
AppleRTCHibernateVars RtcVars;
|
|
BOOLEAN HasHibernateInfo;
|
|
BOOLEAN HasHibernateInfoInRTC;
|
|
UINT8 Index;
|
|
UINT8 *RtcRawVars;
|
|
EFI_DEVICE_PATH_PROTOCOL *BootImagePath;
|
|
EFI_DEVICE_PATH_PROTOCOL *RemainingPath;
|
|
INTN NumPatchedNodes;
|
|
|
|
if (HibernateMask == HIBERNATE_MODE_NONE) {
|
|
return EFI_NOT_FOUND;
|
|
}
|
|
|
|
HasHibernateInfo = FALSE;
|
|
HasHibernateInfoInRTC = FALSE;
|
|
|
|
//
|
|
// If legacy boot-switch-vars exists (NVRAM working), then use it.
|
|
//
|
|
Status = GetVariable2 (L"boot-switch-vars", &gEfiAppleBootGuid, &Value, &Size);
|
|
if (!EFI_ERROR(Status)) {
|
|
//
|
|
// Leave it as is.
|
|
//
|
|
SecureZeroMem (Value, Size);
|
|
FreePool (Value);
|
|
DEBUG ((DEBUG_INFO, "OCB: Found legacy boot-switch-vars\n"));
|
|
return EFI_SUCCESS;
|
|
}
|
|
|
|
Status = GetVariable3 (
|
|
L"boot-image",
|
|
&gEfiAppleBootGuid,
|
|
(VOID **)&BootImagePath,
|
|
&Size,
|
|
&Attributes
|
|
);
|
|
if (!EFI_ERROR(Status)) {
|
|
if (IsDevicePathValid (BootImagePath, Size)) {
|
|
DebugPrintDevicePath (
|
|
DEBUG_INFO,
|
|
"OCB: boot-image pre-fix",
|
|
BootImagePath
|
|
);
|
|
|
|
RemainingPath = BootImagePath;
|
|
NumPatchedNodes = OcFixAppleBootDevicePath (&RemainingPath);
|
|
if (NumPatchedNodes > 0) {
|
|
DebugPrintDevicePath (
|
|
DEBUG_INFO,
|
|
"OCB: boot-image post-fix",
|
|
BootImagePath
|
|
);
|
|
|
|
Status = gRT->SetVariable (
|
|
L"boot-image",
|
|
&gEfiAppleBootGuid,
|
|
Attributes,
|
|
Size,
|
|
BootImagePath
|
|
);
|
|
}
|
|
if (NumPatchedNodes >= 0) {
|
|
DebugPrintDevicePath (
|
|
DEBUG_INFO,
|
|
"OCB: boot-image post-fix remainder",
|
|
RemainingPath
|
|
);
|
|
}
|
|
} else {
|
|
DEBUG ((DEBUG_INFO, "OCB: Invalid boot-image variable\n"));
|
|
}
|
|
|
|
SecureZeroMem (BootImagePath, Size);
|
|
FreePool (BootImagePath);
|
|
}
|
|
|
|
DEBUG ((DEBUG_INFO, "OCB: boot-image is %u bytes - %r\n", (UINT32) Size, Status));
|
|
|
|
RtcRawVars = (UINT8 *) &RtcVars;
|
|
|
|
//
|
|
// Work with RTC memory if allowed.
|
|
//
|
|
if (HibernateMask & HIBERNATE_MODE_RTC) {
|
|
for (Index = 0; Index < sizeof (AppleRTCHibernateVars); Index++) {
|
|
RtcRawVars[Index] = OcRtcRead (Index + 128);
|
|
}
|
|
|
|
HasHibernateInfoInRTC = RtcVars.signature[0] == 'A'
|
|
&& RtcVars.signature[1] == 'A'
|
|
&& RtcVars.signature[2] == 'P'
|
|
&& RtcVars.signature[3] == 'L';
|
|
HasHibernateInfo = HasHibernateInfoInRTC;
|
|
|
|
DEBUG ((DEBUG_INFO, "OCB: RTC hibernation is %d\n", HasHibernateInfoInRTC));
|
|
}
|
|
|
|
if (HibernateMask & HIBERNATE_MODE_NVRAM) {
|
|
//
|
|
// If RTC variables is still written to NVRAM (and RTC is broken).
|
|
// Prior to 10.13.6.
|
|
//
|
|
Status = GetVariable2 (L"IOHibernateRTCVariables", &gEfiAppleBootGuid, &Value, &Size);
|
|
if (!HasHibernateInfo && !EFI_ERROR(Status) && Size == sizeof (RtcVars)) {
|
|
CopyMem (RtcRawVars, Value, sizeof (RtcVars));
|
|
HasHibernateInfo = RtcVars.signature[0] == 'A'
|
|
&& RtcVars.signature[1] == 'A'
|
|
&& RtcVars.signature[2] == 'P'
|
|
&& RtcVars.signature[3] == 'L';
|
|
}
|
|
|
|
DEBUG ((
|
|
DEBUG_INFO,
|
|
"OCB: NVRAM hibernation is %d / %r / %u\n",
|
|
HasHibernateInfo,
|
|
Status,
|
|
(UINT32) Size
|
|
));
|
|
|
|
//
|
|
// Erase RTC variables in NVRAM.
|
|
//
|
|
if (!EFI_ERROR(Status)) {
|
|
Status = gRT->SetVariable (
|
|
L"IOHibernateRTCVariables",
|
|
&gEfiAppleBootGuid,
|
|
EFI_VARIABLE_NON_VOLATILE | EFI_VARIABLE_BOOTSERVICE_ACCESS | EFI_VARIABLE_RUNTIME_ACCESS,
|
|
0,
|
|
NULL
|
|
);
|
|
SecureZeroMem (Value, Size);
|
|
FreePool (Value);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Convert RTC data to boot-key and boot-signature
|
|
//
|
|
if (HasHibernateInfo) {
|
|
gRT->SetVariable (
|
|
L"boot-image-key",
|
|
&gEfiAppleBootGuid,
|
|
EFI_VARIABLE_BOOTSERVICE_ACCESS,
|
|
sizeof (RtcVars.wiredCryptKey),
|
|
RtcVars.wiredCryptKey
|
|
);
|
|
gRT->SetVariable (
|
|
L"boot-signature",
|
|
&gEfiAppleBootGuid,
|
|
EFI_VARIABLE_BOOTSERVICE_ACCESS,
|
|
sizeof (RtcVars.booterSignature),
|
|
RtcVars.booterSignature
|
|
);
|
|
}
|
|
|
|
//
|
|
// Erase RTC memory similarly to AppleBds.
|
|
//
|
|
if (HasHibernateInfoInRTC) {
|
|
SecureZeroMem (RtcRawVars, sizeof(AppleRTCHibernateVars));
|
|
RtcVars.signature[0] = 'D';
|
|
RtcVars.signature[1] = 'E';
|
|
RtcVars.signature[2] = 'A';
|
|
RtcVars.signature[3] = 'D';
|
|
|
|
for (Index = 0; Index < sizeof(AppleRTCHibernateVars); Index++) {
|
|
OcRtcWrite (Index + 128, RtcRawVars[Index]);
|
|
}
|
|
}
|
|
|
|
//
|
|
// We have everything we need now.
|
|
//
|
|
if (HasHibernateInfo) {
|
|
return EFI_SUCCESS;
|
|
}
|
|
|
|
return EFI_NOT_FOUND;
|
|
}
|
|
|
|
BOOLEAN
|
|
OcIsAppleHibernateWake (
|
|
VOID
|
|
)
|
|
{
|
|
EFI_STATUS Status;
|
|
UINTN ValueSize;
|
|
|
|
//
|
|
// This is reverse engineered from boot.efi.
|
|
// To cancel hibernate wake it is enough to delete the variables.
|
|
// Starting with 10.13.6 boot-switch-vars is no longer supported.
|
|
//
|
|
ValueSize = 0;
|
|
Status = gRT->GetVariable (
|
|
L"boot-signature",
|
|
&gEfiAppleBootGuid,
|
|
NULL,
|
|
&ValueSize,
|
|
NULL
|
|
);
|
|
|
|
if (Status == EFI_BUFFER_TOO_SMALL) {
|
|
ValueSize = 0;
|
|
Status = gRT->GetVariable (
|
|
L"boot-image-key",
|
|
&gEfiAppleBootGuid,
|
|
NULL,
|
|
&ValueSize,
|
|
NULL
|
|
);
|
|
|
|
if (Status == EFI_BUFFER_TOO_SMALL) {
|
|
return TRUE;
|
|
}
|
|
} else {
|
|
ValueSize = 0;
|
|
Status = gRT->GetVariable (
|
|
L"boot-switch-vars",
|
|
&gEfiAppleBootGuid,
|
|
NULL,
|
|
&ValueSize,
|
|
NULL
|
|
);
|
|
|
|
if (Status == EFI_BUFFER_TOO_SMALL) {
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|