blob: f83257a9898169b38753321dcdb256899614e59d [file] [log] [blame]
/*
* Copyright (c) 2009, Google Inc.
* All rights reserved.
*
* Copyright (c) 2009-2020, The Linux Foundation. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* * Neither the name of The Linux Foundation nor
* the names of its contributors may be used to endorse or promote
* products derived from this software without specific prior written
* permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NON-INFRINGEMENT ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
*/
#include <Library/DeviceInfo.h>
#include <Library/DrawUI.h>
#include <Library/PartitionTableUpdate.h>
#include <Library/ShutdownServices.h>
#include <Library/VerifiedBootMenu.h>
#include <Library/HypervisorMvCalls.h>
#include <Protocol/EFIMdtp.h>
#include <Protocol/EFIScmModeSwitch.h>
#include <libufdt_sysdeps.h>
#include "AutoGen.h"
#include "BootImage.h"
#include "BootLinux.h"
#include "BootStats.h"
#include "UpdateDeviceTree.h"
#include "libfdt.h"
#include <ufdt_overlay.h>
STATIC QCOM_SCM_MODE_SWITCH_PROTOCOL *pQcomScmModeSwitchProtocol = NULL;
STATIC BOOLEAN BootDevImage;
/* To set load addresses, callers should make sure to initialize the
* BootParamlistPtr before calling this function */
UINT64 SetandGetLoadAddr (BootParamlist *BootParamlistPtr, AddrType Type)
{
STATIC UINT64 KernelLoadAddr;
STATIC UINT64 RamdiskLoadAddr;
if (BootParamlistPtr) {
KernelLoadAddr = BootParamlistPtr->KernelLoadAddr;
RamdiskLoadAddr = BootParamlistPtr->RamdiskLoadAddr;
} else {
switch (Type) {
case LOAD_ADDR_KERNEL:
return KernelLoadAddr;
break;
case LOAD_ADDR_RAMDISK:
return RamdiskLoadAddr;
break;
default:
DEBUG ((EFI_D_ERROR, "Invalid Type to GetLoadAddr():%d\n",
Type));
break;
}
}
return 0;
}
STATIC BOOLEAN
QueryBootParams (UINT64 *KernelLoadAddr, UINT64 *KernelSizeReserved)
{
EFI_STATUS Status;
EFI_STATUS SizeStatus;
UINTN DataSize = 0;
DataSize = sizeof (*KernelLoadAddr);
Status = gRT->GetVariable ((CHAR16 *)L"KernelBaseAddr", &gQcomTokenSpaceGuid,
NULL, &DataSize, KernelLoadAddr);
DataSize = sizeof (*KernelSizeReserved);
SizeStatus = gRT->GetVariable ((CHAR16 *)L"KernelSize", &gQcomTokenSpaceGuid,
NULL, &DataSize, KernelSizeReserved);
return (Status == EFI_SUCCESS &&
SizeStatus == EFI_SUCCESS);
}
STATIC EFI_STATUS
UpdateBootParams (BootParamlist *BootParamlistPtr)
{
UINT64 KernelSizeReserved;
UINT64 KernelLoadAddr;
if (BootParamlistPtr == NULL ) {
DEBUG ((EFI_D_ERROR, "Invalid input parameters\n"));
return EFI_INVALID_PARAMETER;
}
/* The three regions Kernel, Ramdisk and DT should be reserved in memory map
* Query the kernel load address and size from UEFI core, if it's not
* successful use the predefined load addresses */
if (QueryBootParams (&KernelLoadAddr, &KernelSizeReserved)) {
BootParamlistPtr->KernelLoadAddr = KernelLoadAddr;
if (BootParamlistPtr->BootingWith32BitKernel) {
BootParamlistPtr->KernelLoadAddr += KERNEL_32BIT_LOAD_OFFSET;
} else {
BootParamlistPtr->KernelLoadAddr += KERNEL_64BIT_LOAD_OFFSET;
}
BootParamlistPtr->KernelEndAddr = KernelLoadAddr + KernelSizeReserved;
} else {
DEBUG ((EFI_D_VERBOSE, "QueryBootParams Failed: "));
/* If Query of boot params fails, RamdiskEndAddress is end of the
kernel buffer we have. Using same as size of total available buffer,
for relocation of kernel */
if (BootParamlistPtr->BootingWith32BitKernel) {
/* For 32-bit Not all memory is accessible as defined by
RamdiskEndAddress. Using pre-defined offset for backward
compatability */
BootParamlistPtr->KernelLoadAddr =
(EFI_PHYSICAL_ADDRESS) (BootParamlistPtr->BaseMemory |
PcdGet32 (KernelLoadAddress32));
KernelSizeReserved = PcdGet32 (RamdiskEndAddress32);
} else {
BootParamlistPtr->KernelLoadAddr =
(EFI_PHYSICAL_ADDRESS) (BootParamlistPtr->BaseMemory |
PcdGet32 (KernelLoadAddress));
KernelSizeReserved = PcdGet32 (RamdiskEndAddress);
}
BootParamlistPtr->KernelEndAddr = BootParamlistPtr->BaseMemory +
KernelSizeReserved;
DEBUG ((EFI_D_VERBOSE, "calculating dynamic offsets\n"));
}
/* Allocate buffer for ramdisk and tags area, based on ramdisk actual size
and DT maximum supported size. This allows best possible utilization
of buffer for kernel relocation and take care of dynamic change in size
of ramdisk. Add pagesize as a buffer space */
BootParamlistPtr->RamdiskLoadAddr = (BootParamlistPtr->KernelEndAddr -
(LOCAL_ROUND_TO_PAGE (
BootParamlistPtr->RamdiskSize +
BootParamlistPtr->VendorRamdiskSize,
BootParamlistPtr->PageSize) +
BootParamlistPtr->PageSize));
BootParamlistPtr->DeviceTreeLoadAddr = (BootParamlistPtr->RamdiskLoadAddr -
(DT_SIZE_2MB +
BootParamlistPtr->PageSize));
if (BootParamlistPtr->DeviceTreeLoadAddr <=
BootParamlistPtr->KernelLoadAddr) {
DEBUG ((EFI_D_ERROR, "Not Enough space left to load kernel image\n"));
return EFI_BUFFER_TOO_SMALL;
}
return EFI_SUCCESS;
}
STATIC EFI_STATUS
SwitchTo32bitModeBooting (UINT64 KernelLoadAddr, UINT64 DeviceTreeLoadAddr)
{
EFI_STATUS Status;
EFI_HLOS_BOOT_ARGS HlosBootArgs;
SetMem ((VOID *)&HlosBootArgs, sizeof (HlosBootArgs), 0);
HlosBootArgs.el1_x2 = DeviceTreeLoadAddr;
/* Write 0 into el1_x4 to switch to 32bit mode */
HlosBootArgs.el1_x4 = 0;
HlosBootArgs.el1_elr = KernelLoadAddr;
Status = pQcomScmModeSwitchProtocol->SwitchTo32bitMode (HlosBootArgs);
if (EFI_ERROR (Status)) {
DEBUG ((EFI_D_ERROR, "ERROR: Failed to switch to 32 bit mode.Status= %r\n",
Status));
return Status;
}
/*Return Unsupported if the execution ever reaches here*/
return EFI_NOT_STARTED;
}
STATIC EFI_STATUS
UpdateKernelModeAndPkg (BootParamlist *BootParamlistPtr)
{
Kernel64Hdr *Kptr = NULL;
if (BootParamlistPtr == NULL ) {
DEBUG ((EFI_D_ERROR, "Invalid input parameters\n"));
return EFI_INVALID_PARAMETER;
}
BootParamlistPtr->BootingWith32BitKernel = FALSE;
Kptr = (Kernel64Hdr *) (BootParamlistPtr->ImageBuffer +
BootParamlistPtr->PageSize);
if (is_gzip_package ((BootParamlistPtr->ImageBuffer +
BootParamlistPtr->PageSize), BootParamlistPtr->KernelSize)) {
BootParamlistPtr->BootingWithGzipPkgKernel = TRUE;
}
else {
if (!AsciiStrnCmp ((CHAR8 *) Kptr, PATCHED_KERNEL_MAGIC,
sizeof (PATCHED_KERNEL_MAGIC) - 1)) {
BootParamlistPtr->BootingWithPatchedKernel = TRUE;
Kptr = (struct kernel64_hdr *)((VOID *)Kptr +
PATCHED_KERNEL_HEADER_SIZE);
}
if (Kptr->magic_64 != KERNEL64_HDR_MAGIC) {
BootParamlistPtr->BootingWith32BitKernel = TRUE;
}
}
return EFI_SUCCESS;
}
STATIC EFI_STATUS
CheckMDTPStatus (CHAR16 *PartitionName, BootInfo *Info)
{
EFI_STATUS Status = EFI_SUCCESS;
BOOLEAN MdtpActive = FALSE;
CHAR8 StrPartition[MAX_GPT_NAME_SIZE];
CHAR8 PartitionNameAscii[MAX_GPT_NAME_SIZE];
UINT32 PartitionNameLen;
QCOM_MDTP_PROTOCOL *MdtpProtocol;
MDTP_VB_EXTERNAL_PARTITION ExternalPartition;
SetMem ((VOID *)StrPartition, MAX_GPT_NAME_SIZE, 0);
SetMem ((VOID *)PartitionNameAscii, MAX_GPT_NAME_SIZE, 0);
if (FixedPcdGetBool (EnableMdtpSupport)) {
Status = IsMdtpActive (&MdtpActive);
if (EFI_ERROR (Status)) {
DEBUG ((EFI_D_ERROR, "Failed to get activation state for MDTP, "
"Status=%r. Considering MDTP as active and "
"continuing \n",
Status));
if (Status != EFI_NOT_FOUND) {
MdtpActive = TRUE;
}
}
if (MdtpActive) {
/* If MDTP is Active and Dm-Verity Mode is not Enforcing, Block */
if (!IsEnforcing ()) {
DEBUG ((EFI_D_ERROR,
"ERROR: MDTP is active and verity mode is not enforcing \n"));
return EFI_NOT_STARTED;
}
/* If MDTP is Active and Device is in unlocked State, Block */
if (IsUnlocked ()) {
DEBUG ((EFI_D_ERROR,
"ERROR: MDTP is active and DEVICE is unlocked \n"));
return EFI_NOT_STARTED;
}
}
}
UnicodeStrToAsciiStr (PartitionName, PartitionNameAscii);
PartitionNameLen = AsciiStrLen (PartitionNameAscii);
if (Info->MultiSlotBoot)
PartitionNameLen -= (MAX_SLOT_SUFFIX_SZ - 1);
AsciiStrnCpyS (StrPartition, MAX_GPT_NAME_SIZE, "/", AsciiStrLen ("/"));
AsciiStrnCatS (StrPartition, MAX_GPT_NAME_SIZE, PartitionNameAscii,
PartitionNameLen);
if (FixedPcdGetBool (EnableMdtpSupport)) {
Status = gBS->LocateProtocol (&gQcomMdtpProtocolGuid, NULL,
(VOID **)&MdtpProtocol);
if (Status != EFI_NOT_FOUND) {
if (EFI_ERROR (Status)) {
DEBUG ((EFI_D_ERROR, "Failed in locating MDTP protocol, Status=%r\n",
Status));
return Status;
}
AsciiStrnCpyS (ExternalPartition.PartitionName, MAX_PARTITION_NAME_LEN,
StrPartition, AsciiStrLen (StrPartition));
Status = MdtpProtocol->MdtpBootState (MdtpProtocol, &ExternalPartition);
if (EFI_ERROR (Status)) {
/* MdtpVerify should always handle errors internally, so when returned
* back to the caller,
* the return value is expected to be success only.
* Therfore, we don't expect any error status here. */
DEBUG ((EFI_D_ERROR, "MDTP verification failed, Status=%r\n", Status));
return Status;
}
}
else
DEBUG (
(EFI_D_ERROR, "Failed to locate MDTP protocol, Status=%r\n", Status));
}
return Status;
}
STATIC EFI_STATUS
ApplyOverlay (BootParamlist *BootParamlistPtr,
VOID *AppendedDtHdr,
struct fdt_entry_node *DtsList)
{
VOID *FinalDtbHdr = AppendedDtHdr;
VOID *TmpDtbHdr = NULL;
UINT64 ApplyDTStartTime = GetTimerCountms ();
if (BootParamlistPtr == NULL ||
AppendedDtHdr == NULL) {
DEBUG ((EFI_D_ERROR, "ApplyOverlay: Invalid input parameters\n"));
return EFI_INVALID_PARAMETER;
}
if (DtsList == NULL) {
DEBUG ((EFI_D_VERBOSE, "ApplyOverlay: Overlay DT is NULL\n"));
goto out;
}
if (!pre_overlay_malloc ()) {
DEBUG ((EFI_D_ERROR,
"ApplyOverlay: Unable to Allocate Pre Buffer for Overlay\n"));
return EFI_OUT_OF_RESOURCES;
}
TmpDtbHdr = ufdt_install_blob (AppendedDtHdr, fdt_totalsize (AppendedDtHdr));
if (!TmpDtbHdr) {
DEBUG ((EFI_D_ERROR, "ApplyOverlay: Install blob failed\n"));
return EFI_NOT_FOUND;
}
FinalDtbHdr = ufdt_apply_multi_overlay (TmpDtbHdr,
fdt_totalsize (TmpDtbHdr),
DtsList);
DeleteDtList (&DtsList);
if (!FinalDtbHdr) {
DEBUG ((EFI_D_ERROR, "ApplyOverlay: ufdt apply overlay failed\n"));
return EFI_NOT_FOUND;
}
out:
if ((BootParamlistPtr->RamdiskLoadAddr -
BootParamlistPtr->DeviceTreeLoadAddr) <
fdt_totalsize (FinalDtbHdr)) {
DEBUG ((EFI_D_ERROR,
"ApplyOverlay: After overlay DTB size exceeded than supported\n"));
return EFI_UNSUPPORTED;
}
/* If DeviceTreeLoadAddr == AppendedDtHdr
CopyMem will not copy Source Buffer to Destination Buffer
and return Destination BUffer.
*/
gBS->CopyMem ((VOID *)BootParamlistPtr->DeviceTreeLoadAddr,
FinalDtbHdr,
fdt_totalsize (FinalDtbHdr));
post_overlay_free ();
DEBUG ((EFI_D_INFO, "Apply Overlay total time: %lu ms \n",
GetTimerCountms () - ApplyDTStartTime));
return EFI_SUCCESS;
}
STATIC UINT32
GetNumberOfPages (UINT32 ImageSize, UINT32 PageSize)
{
return (ImageSize + PageSize - 1) / PageSize;
}
STATIC EFI_STATUS
DTBImgCheckAndAppendDT (BootInfo *Info, BootParamlist *BootParamlistPtr)
{
VOID *SingleDtHdr = NULL;
VOID *NextDtHdr = NULL;
VOID *BoardDtb = NULL;
VOID *SocDtb = NULL;
VOID *OverrideDtb = NULL;
VOID *Dtb;
BOOLEAN DtboCheckNeeded = FALSE;
BOOLEAN DtboImgInvalid = FALSE;
struct fdt_entry_node *DtsList = NULL;
EFI_STATUS Status;
UINT32 HeaderVersion = 0;
struct boot_img_hdr_v1 *BootImgHdrV1;
struct boot_img_hdr_v2 *BootImgHdrV2;
vendor_boot_img_hdr_v3 *VendorBootImgHdrV3;
UINT32 NumHeaderPages;
UINT32 NumKernelPages;
UINT32 NumSecondPages;
UINT32 NumRamdiskPages;
UINT32 NumVendorRamdiskPages;
UINT32 NumRecoveryDtboPages;
VOID* ImageBuffer = NULL;
UINT32 ImageSize = 0;
if (Info == NULL ||
BootParamlistPtr == NULL) {
DEBUG ((EFI_D_ERROR, "Invalid input parameters\n"));
return EFI_INVALID_PARAMETER;
}
ImageBuffer = BootParamlistPtr->ImageBuffer +
BootParamlistPtr->PageSize +
BootParamlistPtr->PatchedKernelHdrSize;
ImageSize = BootParamlistPtr->KernelSize;
HeaderVersion = Info->HeaderVersion;
if (HeaderVersion > BOOT_HEADER_VERSION_ONE) {
BootImgHdrV1 = (struct boot_img_hdr_v1 *)
((UINT64) BootParamlistPtr->ImageBuffer +
BOOT_IMAGE_HEADER_V1_RECOVERY_DTBO_SIZE_OFFSET);
BootImgHdrV2 = (struct boot_img_hdr_v2 *)
((UINT64) BootParamlistPtr->ImageBuffer +
BOOT_IMAGE_HEADER_V1_RECOVERY_DTBO_SIZE_OFFSET +
BOOT_IMAGE_HEADER_V2_OFFSET);
NumHeaderPages = 1;
NumKernelPages =
GetNumberOfPages (BootParamlistPtr->KernelSize,
BootParamlistPtr->PageSize);
NumRamdiskPages =
GetNumberOfPages (BootParamlistPtr->RamdiskSize,
BootParamlistPtr->PageSize);
NumSecondPages =
GetNumberOfPages (BootParamlistPtr->SecondSize,
BootParamlistPtr->PageSize);
if (HeaderVersion == BOOT_HEADER_VERSION_TWO) {
NumRecoveryDtboPages =
GetNumberOfPages (BootImgHdrV1->recovery_dtbo_size,
BootParamlistPtr->PageSize);
BootParamlistPtr->DtbOffset = BootParamlistPtr->PageSize *
(NumHeaderPages + NumKernelPages + NumRamdiskPages +
NumSecondPages + NumRecoveryDtboPages);
ImageSize = BootImgHdrV2->dtb_size + BootParamlistPtr->DtbOffset;
ImageBuffer = BootParamlistPtr->ImageBuffer;
} else {
VendorBootImgHdrV3 = BootParamlistPtr->VendorImageBuffer;
NumVendorRamdiskPages = GetNumberOfPages (
BootParamlistPtr->VendorRamdiskSize,
BootParamlistPtr->PageSize);
BootParamlistPtr->DtbOffset = BootParamlistPtr->PageSize *
(NumHeaderPages + NumVendorRamdiskPages);
ImageSize = VendorBootImgHdrV3->dtb_size +
BootParamlistPtr->DtbOffset;
// DTB is a part of vendor_boot image
ImageBuffer = BootParamlistPtr->VendorImageBuffer;
}
}
DtboImgInvalid = LoadAndValidateDtboImg (Info, BootParamlistPtr);
if (!DtboImgInvalid) {
// appended device tree
Dtb = DeviceTreeAppended (ImageBuffer,
ImageSize,
BootParamlistPtr->DtbOffset,
(VOID *)BootParamlistPtr->DeviceTreeLoadAddr);
if (!Dtb) {
if (BootParamlistPtr->DtbOffset >= ImageSize) {
DEBUG ((EFI_D_ERROR, "Dtb offset goes beyond the image size\n"));
return EFI_BAD_BUFFER_SIZE;
}
SingleDtHdr = (BootParamlistPtr->ImageBuffer +
BootParamlistPtr->PageSize +
BootParamlistPtr->DtbOffset);
if (!fdt_check_header (SingleDtHdr)) {
if ((ImageSize - BootParamlistPtr->DtbOffset) <
fdt_totalsize (SingleDtHdr)) {
DEBUG ((EFI_D_ERROR, "Dtb offset goes beyond the image size\n"));
return EFI_BAD_BUFFER_SIZE;
}
NextDtHdr =
(VOID *)((uintptr_t)SingleDtHdr + fdt_totalsize (SingleDtHdr));
if (!fdt_check_header (NextDtHdr)) {
DEBUG ((EFI_D_VERBOSE, "Not the single appended DTB\n"));
return EFI_NOT_FOUND;
}
DEBUG ((EFI_D_VERBOSE, "Single appended DTB found\n"));
if (CHECK_ADD64 (BootParamlistPtr->DeviceTreeLoadAddr,
fdt_totalsize (SingleDtHdr))) {
DEBUG ((EFI_D_ERROR,
"Integer Overflow: in single dtb header addition\n"));
return EFI_BAD_BUFFER_SIZE;
}
gBS->CopyMem ((VOID *)BootParamlistPtr->DeviceTreeLoadAddr,
SingleDtHdr, fdt_totalsize (SingleDtHdr));
} else {
DEBUG ((EFI_D_ERROR, "Error: Device Tree blob not found\n"));
return EFI_NOT_FOUND;
}
}
} else {
/*It is the case of DTB overlay Get the Soc specific dtb */
SocDtb = GetSocDtb (ImageBuffer,
ImageSize,
BootParamlistPtr->DtbOffset,
(VOID *)BootParamlistPtr->DeviceTreeLoadAddr);
if (!SocDtb) {
DEBUG ((EFI_D_ERROR,
"Error: Appended Soc Device Tree blob not found\n"));
return EFI_NOT_FOUND;
}
/*Check do we really need to gothrough DTBO or not*/
DtboCheckNeeded = GetDtboNeeded ();
if (DtboCheckNeeded == TRUE) {
BoardDtb = GetBoardDtb (Info, BootParamlistPtr->DtboImgBuffer);
if (!BoardDtb) {
DEBUG ((EFI_D_ERROR, "Error: Board Dtbo blob not found\n"));
return EFI_NOT_FOUND;
}
if (!AppendToDtList (&DtsList,
(fdt64_t)BoardDtb,
fdt_totalsize (BoardDtb))) {
DEBUG ((EFI_D_ERROR,
"Unable to Allocate buffer for Overlay DT\n"));
DeleteDtList (&DtsList);
return EFI_OUT_OF_RESOURCES;
}
}
/* If hypervisor boot info is present, append dtbo info passed from hyp */
if (IsVmEnabled ()) {
if (BootParamlistPtr->HypDtboBaseAddr == NULL) {
DEBUG ((EFI_D_ERROR, "Error: HypOverlay DT is NULL\n"));
return EFI_NOT_FOUND;
}
for (UINT32 i = 0; i < BootParamlistPtr->NumHypDtbos; i++) {
/* Flag the invalid dtbos and overlay the valid ones */
if (!BootParamlistPtr->HypDtboBaseAddr[i] ||
fdt_check_header ((VOID *)BootParamlistPtr->HypDtboBaseAddr[i])) {
DEBUG ((EFI_D_ERROR, "HypInfo: Not overlaying hyp dtbo"
"Dtbo :%d is null or Bad DT header\n", i));
continue;
}
if (!AppendToDtList (&DtsList,
(fdt64_t)BootParamlistPtr->HypDtboBaseAddr[i],
fdt_totalsize (BootParamlistPtr->HypDtboBaseAddr[i]))) {
DEBUG ((EFI_D_ERROR,
"Unable to Allocate buffer for HypOverlay DT num: %d\n", i));
DeleteDtList (&DtsList);
return EFI_OUT_OF_RESOURCES;
}
}
}
// Only enabled to debug builds.
if (!TargetBuildVariantUser ()) {
Status = GetOvrdDtb (&OverrideDtb);
if (Status == EFI_SUCCESS &&
OverrideDtb &&
!AppendToDtList (&DtsList,
(fdt64_t)OverrideDtb,
fdt_totalsize (OverrideDtb))) {
DEBUG ((EFI_D_ERROR,
"Unable to allocate buffer for Override DT\n"));
DeleteDtList (&DtsList);
return EFI_OUT_OF_RESOURCES;
}
}
Status = ApplyOverlay (BootParamlistPtr,
SocDtb,
DtsList);
if (Status != EFI_SUCCESS) {
DEBUG ((EFI_D_ERROR, "Error: Dtb overlay failed\n"));
return Status;
}
}
return EFI_SUCCESS;
}
STATIC EFI_STATUS
GZipPkgCheck (BootParamlist *BootParamlistPtr)
{
UINT32 OutLen = 0;
UINT64 OutAvaiLen = 0;
struct kernel64_hdr *Kptr = NULL;
UINT64 DecompressStartTime;
if (BootParamlistPtr == NULL) {
DEBUG ((EFI_D_ERROR, "Invalid input parameters\n"));
return EFI_INVALID_PARAMETER;
}
if (BootParamlistPtr->BootingWithGzipPkgKernel) {
OutAvaiLen = BootParamlistPtr->DeviceTreeLoadAddr -
BootParamlistPtr->KernelLoadAddr;
if (OutAvaiLen > MAX_UINT32) {
DEBUG ((EFI_D_ERROR,
"Integer Overflow: the length of decompressed data = %u\n",
OutAvaiLen));
return EFI_BAD_BUFFER_SIZE;
}
DecompressStartTime = GetTimerCountms ();
if (decompress (
(UINT8 *)(BootParamlistPtr->ImageBuffer +
BootParamlistPtr->PageSize), // Read blob using BlockIo
BootParamlistPtr->KernelSize, // Blob size
(UINT8 *)BootParamlistPtr->KernelLoadAddr, // Load address, allocated
(UINT32)OutAvaiLen, // Allocated Size
&BootParamlistPtr->DtbOffset, &OutLen)) {
DEBUG ((EFI_D_ERROR, "Decompressing kernel image failed!!!\n"));
return RETURN_OUT_OF_RESOURCES;
}
if (OutLen <= sizeof (struct kernel64_hdr *)) {
DEBUG ((EFI_D_ERROR,
"Decompress kernel size is smaller than image header size\n"));
return RETURN_OUT_OF_RESOURCES;
}
Kptr = (Kernel64Hdr *) BootParamlistPtr->KernelLoadAddr;
DEBUG ((EFI_D_INFO, "Decompressing kernel image total time: %lu ms\n",
GetTimerCountms () - DecompressStartTime));
} else {
Kptr = (struct kernel64_hdr *)(BootParamlistPtr->ImageBuffer
+ BootParamlistPtr->PageSize);
/* Patch kernel support only for 64-bit */
if (BootParamlistPtr->BootingWithPatchedKernel) {
DEBUG ((EFI_D_VERBOSE, "Patched kernel detected\n"));
/* The size of the kernel is stored at start of kernel image + 16
* The dtb would start just after the kernel */
gBS->CopyMem ((VOID *)&BootParamlistPtr->DtbOffset,
(VOID *) (BootParamlistPtr->ImageBuffer +
BootParamlistPtr->PageSize +
sizeof (PATCHED_KERNEL_MAGIC) - 1),
sizeof (BootParamlistPtr->DtbOffset));
BootParamlistPtr->PatchedKernelHdrSize = PATCHED_KERNEL_HEADER_SIZE;
Kptr = (struct kernel64_hdr *)((VOID *)Kptr +
BootParamlistPtr->PatchedKernelHdrSize);
}
if (Kptr->magic_64 != KERNEL64_HDR_MAGIC) {
if (BootParamlistPtr->KernelSize <=
DTB_OFFSET_LOCATION_IN_ARCH32_KERNEL_HDR) {
DEBUG ((EFI_D_ERROR, "DTB offset goes beyond kernel size.\n"));
return EFI_BAD_BUFFER_SIZE;
}
gBS->CopyMem ((VOID *)&BootParamlistPtr->DtbOffset,
((VOID *)Kptr + DTB_OFFSET_LOCATION_IN_ARCH32_KERNEL_HDR),
sizeof (BootParamlistPtr->DtbOffset));
}
gBS->CopyMem ((VOID *)BootParamlistPtr->KernelLoadAddr, (VOID *)Kptr,
BootParamlistPtr->KernelSize);
}
if (Kptr->magic_64 != KERNEL64_HDR_MAGIC) {
/* For GZipped 32-bit Kernel */
BootParamlistPtr->BootingWith32BitKernel = TRUE;
} else {
if (Kptr->ImageSize >
(BootParamlistPtr->DeviceTreeLoadAddr -
BootParamlistPtr->KernelLoadAddr)) {
DEBUG ((EFI_D_ERROR,
"DTB header can get corrupted due to runtime kernel size\n"));
return RETURN_OUT_OF_RESOURCES;
}
}
return EFI_SUCCESS;
}
STATIC EFI_STATUS
LoadAddrAndDTUpdate (BootInfo *Info, BootParamlist *BootParamlistPtr)
{
EFI_STATUS Status;
UINT64 RamdiskLoadAddr;
UINT64 RamdiskEndAddr = 0;
UINT32 TotalRamdiskSize;
if (BootParamlistPtr == NULL) {
DEBUG ((EFI_D_ERROR, "Invalid input parameters\n"));
return EFI_INVALID_PARAMETER;
}
RamdiskLoadAddr = BootParamlistPtr->RamdiskLoadAddr;
TotalRamdiskSize = BootParamlistPtr->RamdiskSize +
BootParamlistPtr->VendorRamdiskSize;
if (RamdiskEndAddr - RamdiskLoadAddr < TotalRamdiskSize) {
DEBUG ((EFI_D_ERROR, "Error: Ramdisk size is over the limit\n"));
return EFI_BAD_BUFFER_SIZE;
}
if (CHECK_ADD64 ((UINT64)BootParamlistPtr->ImageBuffer,
BootParamlistPtr->RamdiskOffset)) {
DEBUG ((EFI_D_ERROR, "Integer Overflow: ImageBuffer=%u, "
"RamdiskOffset=%u\n",
BootParamlistPtr->ImageBuffer,
BootParamlistPtr->RamdiskOffset));
return EFI_BAD_BUFFER_SIZE;
}
Status = UpdateDeviceTree ((VOID *)BootParamlistPtr->DeviceTreeLoadAddr,
BootParamlistPtr->FinalCmdLine,
(VOID *)RamdiskLoadAddr, TotalRamdiskSize,
BootParamlistPtr->BootingWith32BitKernel);
if (Status != EFI_SUCCESS) {
DEBUG ((EFI_D_ERROR, "Device Tree update failed Status:%r\n", Status));
return Status;
}
/* If the boot-image version is greater than 2, place the vendor-ramdisk
* first in the memory, and then place ramdisk.
* This concatination would result in an overlay for .gzip and .cpio formats.
*/
if (Info->HeaderVersion >= BOOT_HEADER_VERSION_THREE) {
gBS->CopyMem ((VOID *)RamdiskLoadAddr,
BootParamlistPtr->VendorImageBuffer +
BootParamlistPtr->PageSize,
BootParamlistPtr->VendorRamdiskSize);
RamdiskLoadAddr += BootParamlistPtr->VendorRamdiskSize;
}
gBS->CopyMem ((CHAR8 *)RamdiskLoadAddr,
BootParamlistPtr->ImageBuffer +
BootParamlistPtr->RamdiskOffset,
BootParamlistPtr->RamdiskSize);
if (BootParamlistPtr->BootingWith32BitKernel) {
if (CHECK_ADD64 (BootParamlistPtr->KernelLoadAddr,
BootParamlistPtr->KernelSizeActual)) {
DEBUG ((EFI_D_ERROR, "Integer Overflow: while Kernel image copy\n"));
return EFI_BAD_BUFFER_SIZE;
}
if (BootParamlistPtr->KernelLoadAddr +
BootParamlistPtr->KernelSizeActual >
BootParamlistPtr->DeviceTreeLoadAddr) {
DEBUG ((EFI_D_ERROR, "Kernel size is over the limit\n"));
return EFI_INVALID_PARAMETER;
}
gBS->CopyMem ((CHAR8 *)BootParamlistPtr->KernelLoadAddr,
BootParamlistPtr->ImageBuffer +
BootParamlistPtr->PageSize,
BootParamlistPtr->KernelSizeActual);
}
return EFI_SUCCESS;
}
STATIC EFI_STATUS
CatCmdLine (BootParamlist *BootParamlistPtr,
boot_img_hdr_v3 *BootImgHdrV3,
vendor_boot_img_hdr_v3 *VendorBootImgHdrV3)
{
UINTN MaxCmdLineLen = BOOT_ARGS_SIZE +
BOOT_EXTRA_ARGS_SIZE + VENDOR_BOOT_ARGS_SIZE;
BootParamlistPtr->CmdLine = AllocateZeroPool (MaxCmdLineLen);
if (!BootParamlistPtr->CmdLine) {
DEBUG ((EFI_D_ERROR,
"CatCmdLine: Failed to allocate memory for cmdline\n"));
return EFI_OUT_OF_RESOURCES;
}
/* Place the vendor_boot image cmdline first so that the cmdline
* from boot image takes precedence in case of duplicates.
*/
AsciiStrCpyS (BootParamlistPtr->CmdLine, MaxCmdLineLen,
(CONST CHAR8 *)VendorBootImgHdrV3->cmdline);
AsciiStrCatS (BootParamlistPtr->CmdLine, MaxCmdLineLen, " ");
AsciiStrCatS (BootParamlistPtr->CmdLine, MaxCmdLineLen,
(CONST CHAR8 *)BootImgHdrV3->cmdline);
return EFI_SUCCESS;
}
STATIC EFI_STATUS
UpdateBootParamsSizeAndCmdLine (BootInfo *Info, BootParamlist *BootParamlistPtr)
{
EFI_STATUS Status = EFI_SUCCESS;
UINTN VendorBootImgSize;
boot_img_hdr_v3 *BootImgHdrV3;
vendor_boot_img_hdr_v3 *VendorBootImgHdrV3;
if (Info->HeaderVersion < BOOT_HEADER_VERSION_THREE) {
BootParamlistPtr->KernelSize =
((boot_img_hdr *)(BootParamlistPtr->ImageBuffer))->kernel_size;
BootParamlistPtr->RamdiskSize =
((boot_img_hdr *)(BootParamlistPtr->ImageBuffer))->ramdisk_size;
BootParamlistPtr->SecondSize =
((boot_img_hdr *)(BootParamlistPtr->ImageBuffer))->second_size;
BootParamlistPtr->PageSize =
((boot_img_hdr *)(BootParamlistPtr->ImageBuffer))->page_size;
BootParamlistPtr->CmdLine = (CHAR8 *)&(((boot_img_hdr *)
(BootParamlistPtr->ImageBuffer))->cmdline[0]);
BootParamlistPtr->CmdLine[BOOT_ARGS_SIZE - 1] = '\0';
return EFI_SUCCESS;
}
BootImgHdrV3 = BootParamlistPtr->ImageBuffer;
Status = GetImage (Info, (VOID **)&VendorBootImgHdrV3,
&VendorBootImgSize, "vendor_boot");
if (Status != EFI_SUCCESS) {
DEBUG ((EFI_D_ERROR,
"UpdateBootParamsSizeAndCmdLine: Failed to find vendor_boot image\n"));
return Status;
}
BootParamlistPtr->VendorImageBuffer = VendorBootImgHdrV3;
BootParamlistPtr->VendorImageSize = VendorBootImgSize;
BootParamlistPtr->KernelSize = BootImgHdrV3->kernel_size;
BootParamlistPtr->RamdiskSize = BootImgHdrV3->ramdisk_size;
BootParamlistPtr->VendorRamdiskSize =
VendorBootImgHdrV3->vendor_ramdisk_size;
BootParamlistPtr->PageSize = VendorBootImgHdrV3->page_size;
BootParamlistPtr->SecondSize = 0;
Status = CatCmdLine (BootParamlistPtr, BootImgHdrV3, VendorBootImgHdrV3);
if (Status != EFI_SUCCESS) {
DEBUG ((EFI_D_ERROR,
"UpdateBootParamsSizeAndCmdLine: Failed to cat cmdline\n"));
return Status;
}
return EFI_SUCCESS;
}
EFI_STATUS
BootLinux (BootInfo *Info)
{
EFI_STATUS Status;
CHAR16 *PartitionName = NULL;
BOOLEAN Recovery = FALSE;
BOOLEAN AlarmBoot = FALSE;
LINUX_KERNEL LinuxKernel;
LINUX_KERNEL32 LinuxKernel32;
UINT32 RamdiskSizeActual = 0;
UINT32 SecondSizeActual = 0;
/*Boot Image header information variables*/
CHAR8 FfbmStr[FFBM_MODE_BUF_SIZE] = {'\0'};
BOOLEAN IsModeSwitch = FALSE;
BootParamlist BootParamlistPtr = {0};
if (Info == NULL) {
DEBUG ((EFI_D_ERROR, "BootLinux: invalid parameter Info\n"));
return EFI_INVALID_PARAMETER;
}
if (IsVmEnabled ()) {
Status = CheckAndSetVmData (&BootParamlistPtr);
if (Status != EFI_SUCCESS) {
DEBUG ((EFI_D_ERROR, "Failed to update HypData!! Status:%r\n", Status));
return Status;
}
}
PartitionName = Info->Pname;
Recovery = Info->BootIntoRecovery;
AlarmBoot = Info->BootReasonAlarm;
if (!StrnCmp (PartitionName, (CONST CHAR16 *)L"boot",
StrLen ((CONST CHAR16 *)L"boot"))) {
Status = GetFfbmCommand (FfbmStr, FFBM_MODE_BUF_SIZE);
if (Status != EFI_SUCCESS) {
DEBUG ((EFI_D_VERBOSE, "No Ffbm cookie found, ignore: %r\n", Status));
FfbmStr[0] = '\0';
}
}
Status = GetImage (Info,
&BootParamlistPtr.ImageBuffer,
(UINTN *)&BootParamlistPtr.ImageSize,
((!Info->MultiSlotBoot ||
IsDynamicPartitionSupport ()) &&
(Recovery &&
!IsBuildUseRecoveryAsBoot ()))?
"recovery" : "boot");
if (Status != EFI_SUCCESS ||
BootParamlistPtr.ImageBuffer == NULL ||
BootParamlistPtr.ImageSize <= 0) {
DEBUG ((EFI_D_ERROR, "BootLinux: Get%aImage failed!\n",
(!Info->MultiSlotBoot &&
(Recovery &&
!IsBuildUseRecoveryAsBoot ()))? "Recovery" : "Boot"));
return EFI_NOT_STARTED;
}
/* Find if MDTP is enabled and Active */
Status = CheckMDTPStatus (PartitionName, Info);
if (Status != EFI_SUCCESS) {
return Status;
}
Info->HeaderVersion = ((boot_img_hdr *)
(BootParamlistPtr.ImageBuffer))->header_version;
Status = UpdateBootParamsSizeAndCmdLine (Info, &BootParamlistPtr);
if (Status != EFI_SUCCESS) {
return Status;
}
// Retrive Base Memory Address from Ram Partition Table
Status = BaseMem (&BootParamlistPtr.BaseMemory);
if (Status != EFI_SUCCESS) {
DEBUG ((EFI_D_ERROR, "Base memory not found!!! Status:%r\n", Status));
return Status;
}
Status = UpdateKernelModeAndPkg (&BootParamlistPtr);
if (Status != EFI_SUCCESS) {
return Status;
}
Status = UpdateBootParams (&BootParamlistPtr);
if (Status != EFI_SUCCESS) {
return Status;
}
SetandGetLoadAddr (&BootParamlistPtr, LOAD_ADDR_NONE);
Status = GZipPkgCheck (&BootParamlistPtr);
if (Status != EFI_SUCCESS) {
return Status;
}
/*Finds out the location of device tree image and ramdisk image within the
*boot image
*Kernel, Ramdisk and Second sizes all rounded to page
*The offset and the LOCAL_ROUND_TO_PAGE function is written in a way that it
*is done the same in LK*/
BootParamlistPtr.KernelSizeActual = LOCAL_ROUND_TO_PAGE (
BootParamlistPtr.KernelSize,
BootParamlistPtr.PageSize);
RamdiskSizeActual = LOCAL_ROUND_TO_PAGE (BootParamlistPtr.RamdiskSize,
BootParamlistPtr.PageSize);
SecondSizeActual = LOCAL_ROUND_TO_PAGE (BootParamlistPtr.SecondSize,
BootParamlistPtr.PageSize);
/*Offsets are the location of the images within the boot image*/
BootParamlistPtr.RamdiskOffset = ADD_OF (BootParamlistPtr.PageSize,
BootParamlistPtr.KernelSizeActual);
if (!BootParamlistPtr.RamdiskOffset) {
DEBUG ((EFI_D_ERROR, "Integer Overflow: PageSize=%u, KernelSizeActual=%u\n",
BootParamlistPtr.PageSize, BootParamlistPtr.KernelSizeActual));
return EFI_BAD_BUFFER_SIZE;
}
DEBUG ((EFI_D_VERBOSE, "Kernel Load Address: 0x%x\n",
BootParamlistPtr.KernelLoadAddr));
DEBUG ((EFI_D_VERBOSE, "Kernel Size Actual: 0x%x\n",
BootParamlistPtr.KernelSizeActual));
DEBUG ((EFI_D_VERBOSE, "Second Size Actual: 0x%x\n", SecondSizeActual));
DEBUG ((EFI_D_VERBOSE, "Ramdisk Load Address: 0x%x\n",
BootParamlistPtr.RamdiskLoadAddr));
DEBUG ((EFI_D_VERBOSE, "Ramdisk Size Actual: 0x%x\n", RamdiskSizeActual));
DEBUG ((EFI_D_VERBOSE, "Ramdisk Offset: 0x%x\n",
BootParamlistPtr.RamdiskOffset));
DEBUG (
(EFI_D_VERBOSE, "Device Tree Load Address: 0x%x\n",
BootParamlistPtr.DeviceTreeLoadAddr));
if (AsciiStrStr (BootParamlistPtr.CmdLine, "root=")) {
BootDevImage = TRUE;
}
Status = DTBImgCheckAndAppendDT (Info, &BootParamlistPtr);
if (Status != EFI_SUCCESS) {
return Status;
}
/* Updates the command line from boot image, appends device serial no.,
* baseband information, etc.
* Called before ShutdownUefiBootServices as it uses some boot service
* functions
*/
Status = UpdateCmdLine (BootParamlistPtr.CmdLine, FfbmStr, Recovery,
AlarmBoot, Info->VBCmdLine, &BootParamlistPtr.FinalCmdLine,
Info->HeaderVersion);
if (EFI_ERROR (Status)) {
DEBUG ((EFI_D_ERROR, "Error updating cmdline. Device Error %r\n", Status));
return Status;
}
Status = LoadAddrAndDTUpdate (Info, &BootParamlistPtr);
if (Status != EFI_SUCCESS) {
return Status;
}
FreeVerifiedBootResource (Info);
/* Free the boot logo blt buffer before starting kernel */
FreeBootLogoBltBuffer ();
if (BootParamlistPtr.BootingWith32BitKernel) {
Status = gBS->LocateProtocol (&gQcomScmModeSwithProtocolGuid, NULL,
(VOID **)&pQcomScmModeSwitchProtocol);
if (!EFI_ERROR (Status))
IsModeSwitch = TRUE;
}
DEBUG ((EFI_D_INFO, "\nShutting Down UEFI Boot Services: %lu ms\n",
GetTimerCountms ()));
/*Shut down UEFI boot services*/
Status = ShutdownUefiBootServices ();
if (EFI_ERROR (Status)) {
DEBUG ((EFI_D_ERROR,
"ERROR: Can not shutdown UEFI boot services. Status=0x%X\n",
Status));
goto Exit;
}
PreparePlatformHardware ();
BootStatsSetTimeStamp (BS_KERNEL_ENTRY);
if (IsVmEnabled ()) {
DisableHypUartUsageForLogging ();
}
//
// Start the Linux Kernel
//
if (BootParamlistPtr.BootingWith32BitKernel) {
if (IsModeSwitch) {
Status = SwitchTo32bitModeBooting (
(UINT64)BootParamlistPtr.KernelLoadAddr,
(UINT64)BootParamlistPtr.DeviceTreeLoadAddr);
if (EFI_ERROR (Status)) {
goto Exit;
}
}
// Booting into 32 bit kernel.
LinuxKernel32 = (LINUX_KERNEL32) (UINT64)BootParamlistPtr.KernelLoadAddr;
LinuxKernel32 (0, 0, (UINTN)BootParamlistPtr.DeviceTreeLoadAddr);
// Should never reach here. After life support is not available
DEBUG ((EFI_D_ERROR, "After Life support not available\n"));
goto Exit;
}
LinuxKernel = (LINUX_KERNEL) (UINT64)BootParamlistPtr.KernelLoadAddr;
LinuxKernel ((UINT64)BootParamlistPtr.DeviceTreeLoadAddr, 0, 0, 0);
// Kernel should never exit
// After Life services are not provided
Exit:
// Only be here if we fail to start Linux
CpuDeadLoop ();
return EFI_NOT_STARTED;
}
/**
Check image header
@param[in] ImageHdrBuffer Supplies the address where a pointer to the image
header buffer.
@param[in] ImageHdrSize Supplies the address where a pointer to the image
header size.
@param[in] VendorImageHdrBuffer Supplies the address where a pointer to
the image header buffer.
@param[in] VendorImageHdrSize Supplies the address where a pointer to
the image header size.
@param[out] ImageSizeActual The Pointer for image actual size.
@param[out] PageSize The Pointer for page size..
@retval EFI_SUCCESS Check image header successfully.
@retval other Failed to check image header.
**/
EFI_STATUS
CheckImageHeader (VOID *ImageHdrBuffer,
UINT32 ImageHdrSize,
VOID *VendorImageHdrBuffer,
UINT32 VendorImageHdrSize,
UINT32 *ImageSizeActual,
UINT32 *PageSize,
BOOLEAN BootIntoRecovery)
{
EFI_STATUS Status = EFI_SUCCESS;
struct boot_img_hdr_v2 *BootImgHdrV2;
boot_img_hdr_v3 *BootImgHdrV3;
vendor_boot_img_hdr_v3 *VendorBootImgHdrV3;
UINT32 KernelSizeActual = 0;
UINT32 DtSizeActual = 0;
UINT32 RamdiskSizeActual = 0;
UINT32 VendorRamdiskSizeActual = 0;
// Boot Image header information variables
UINT32 HeaderVersion = 0;
UINT32 KernelSize = 0;
UINT32 RamdiskSize = 0;
UINT32 VendorRamdiskSize = 0;
UINT32 SecondSize = 0;
UINT32 DtSize = 0;
UINT32 tempImgSize = 0;
if (CompareMem ((VOID *)((boot_img_hdr *)(ImageHdrBuffer))->magic, BOOT_MAGIC,
BOOT_MAGIC_SIZE)) {
DEBUG ((EFI_D_ERROR, "Invalid boot image header\n"));
return EFI_NO_MEDIA;
}
HeaderVersion = ((boot_img_hdr *)(ImageHdrBuffer))->header_version;
if (HeaderVersion < BOOT_HEADER_VERSION_THREE) {
KernelSize = ((boot_img_hdr *)(ImageHdrBuffer))->kernel_size;
RamdiskSize = ((boot_img_hdr *)(ImageHdrBuffer))->ramdisk_size;
SecondSize = ((boot_img_hdr *)(ImageHdrBuffer))->second_size;
*PageSize = ((boot_img_hdr *)(ImageHdrBuffer))->page_size;
} else {
if (CompareMem ((VOID *)((vendor_boot_img_hdr_v3 *)
(VendorImageHdrBuffer))->magic,
VENDOR_BOOT_MAGIC, VENDOR_BOOT_MAGIC_SIZE)) {
DEBUG ((EFI_D_ERROR, "Invalid vendor_boot image header\n"));
return EFI_NO_MEDIA;
}
BootImgHdrV3 = ImageHdrBuffer;
VendorBootImgHdrV3 = VendorImageHdrBuffer;
KernelSize = BootImgHdrV3->kernel_size;
RamdiskSize = BootImgHdrV3->ramdisk_size;
VendorRamdiskSize = VendorBootImgHdrV3->vendor_ramdisk_size;
*PageSize = VendorBootImgHdrV3->page_size;
DtSize = VendorBootImgHdrV3->dtb_size;
if (*PageSize > BOOT_IMG_MAX_PAGE_SIZE) {
DEBUG ((EFI_D_ERROR, "Invalid vendor-img pagesize. "
"MAX: %u. PageSize: %u and VendorImageHdrSize: %u\n",
BOOT_IMG_MAX_PAGE_SIZE, *PageSize, VendorImageHdrSize));
return EFI_BAD_BUFFER_SIZE;
}
VendorRamdiskSizeActual = ROUND_TO_PAGE (VendorRamdiskSize, *PageSize - 1);
if (VendorRamdiskSize &&
!VendorRamdiskSizeActual) {
DEBUG ((EFI_D_ERROR, "Integer Overflow: Vendor Ramdisk Size = %u\n",
RamdiskSize));
return EFI_BAD_BUFFER_SIZE;
}
}
if (!KernelSize || !*PageSize) {
DEBUG ((EFI_D_ERROR, "Invalid image Sizes\n"));
DEBUG (
(EFI_D_ERROR, "KernelSize: %u, PageSize=%u\n", KernelSize, *PageSize));
return EFI_BAD_BUFFER_SIZE;
}
if ((*PageSize != ImageHdrSize) && (*PageSize > BOOT_IMG_MAX_PAGE_SIZE)) {
DEBUG ((EFI_D_ERROR, "Invalid image pagesize\n"));
DEBUG ((EFI_D_ERROR, "MAX: %u. PageSize: %u and ImageHdrSize: %u\n",
BOOT_IMG_MAX_PAGE_SIZE, *PageSize, ImageHdrSize));
return EFI_BAD_BUFFER_SIZE;
}
KernelSizeActual = ROUND_TO_PAGE (KernelSize, *PageSize - 1);
if (!KernelSizeActual) {
DEBUG ((EFI_D_ERROR, "Integer Overflow: Kernel Size = %u\n", KernelSize));
return EFI_BAD_BUFFER_SIZE;
}
RamdiskSizeActual = ROUND_TO_PAGE (RamdiskSize, *PageSize - 1);
if (RamdiskSize && !RamdiskSizeActual) {
DEBUG ((EFI_D_ERROR, "Integer Overflow: Ramdisk Size = %u\n", RamdiskSize));
return EFI_BAD_BUFFER_SIZE;
}
if (HeaderVersion == BOOT_HEADER_VERSION_TWO) {
BootImgHdrV2 = (struct boot_img_hdr_v2 *)
((UINT64) ImageHdrBuffer +
BOOT_IMAGE_HEADER_V1_RECOVERY_DTBO_SIZE_OFFSET +
BOOT_IMAGE_HEADER_V2_OFFSET);
DtSize = BootImgHdrV2->dtb_size;
}
// DT size doesn't apply to header versions 0 and 1
if (HeaderVersion >= BOOT_HEADER_VERSION_TWO) {
DtSizeActual = ROUND_TO_PAGE (DtSize, *PageSize - 1);
if (DtSize &&
!DtSizeActual) {
DEBUG ((EFI_D_ERROR, "Integer Overflow: dt Size = %u\n", DtSize));
return EFI_BAD_BUFFER_SIZE;
}
}
*ImageSizeActual = ADD_OF (*PageSize, KernelSizeActual);
if (!*ImageSizeActual) {
DEBUG ((EFI_D_ERROR, "Integer Overflow: Actual Kernel size = %u\n",
KernelSizeActual));
return EFI_BAD_BUFFER_SIZE;
}
tempImgSize = *ImageSizeActual;
*ImageSizeActual = ADD_OF (*ImageSizeActual, RamdiskSizeActual);
if (!*ImageSizeActual) {
DEBUG ((EFI_D_ERROR,
"Integer Overflow: ImgSizeActual=%u, RamdiskActual=%u\n",
tempImgSize, RamdiskSizeActual));
return EFI_BAD_BUFFER_SIZE;
}
tempImgSize = *ImageSizeActual;
/*
* As the DTB is not not a part of boot-images with header versions greater
* than two, ignore considering its size for calculating the total image size
*/
if (HeaderVersion < BOOT_HEADER_VERSION_THREE) {
*ImageSizeActual = ADD_OF (*ImageSizeActual, DtSizeActual);
if (!*ImageSizeActual) {
DEBUG ((EFI_D_ERROR, "Integer Overflow: ImgSizeActual=%u,"
" DtSizeActual=%u\n", tempImgSize, DtSizeActual));
return EFI_BAD_BUFFER_SIZE;
}
}
if (BootIntoRecovery &&
HeaderVersion > BOOT_HEADER_VERSION_ZERO &&
HeaderVersion < BOOT_HEADER_VERSION_THREE) {
struct boot_img_hdr_v1 *Hdr1 =
(struct boot_img_hdr_v1 *) (ImageHdrBuffer + sizeof (boot_img_hdr));
UINT32 RecoveryDtboActual = 0;
if (HeaderVersion == BOOT_HEADER_VERSION_ONE) {
if ((Hdr1->header_size !=
sizeof (struct boot_img_hdr_v1) + sizeof (boot_img_hdr))) {
DEBUG ((EFI_D_ERROR,
"Invalid boot image header: %d\n", Hdr1->header_size));
return EFI_BAD_BUFFER_SIZE;
}
}
else {
if ((Hdr1->header_size !=
BOOT_IMAGE_HEADER_V1_RECOVERY_DTBO_SIZE_OFFSET +
BOOT_IMAGE_HEADER_V2_OFFSET +
sizeof (struct boot_img_hdr_v2))) {
DEBUG ((EFI_D_ERROR,
"Invalid boot image header: %d\n", Hdr1->header_size));
return EFI_BAD_BUFFER_SIZE;
}
}
RecoveryDtboActual = ROUND_TO_PAGE (Hdr1->recovery_dtbo_size,
*PageSize - 1);
if (RecoveryDtboActual > DTBO_MAX_SIZE_ALLOWED) {
DEBUG ((EFI_D_ERROR, "Recovery Dtbo Size too big %x, Allowed size %x\n",
RecoveryDtboActual, DTBO_MAX_SIZE_ALLOWED));
return EFI_BAD_BUFFER_SIZE;
}
if (CHECK_ADD64 (Hdr1->recovery_dtbo_offset, RecoveryDtboActual)) {
DEBUG ((EFI_D_ERROR, "Integer Overflow: RecoveryDtboOffset=%u "
"RecoveryDtboActual=%u\n",
Hdr1->recovery_dtbo_offset, RecoveryDtboActual));
return EFI_BAD_BUFFER_SIZE;
}
tempImgSize = *ImageSizeActual;
*ImageSizeActual = ADD_OF (*ImageSizeActual, RecoveryDtboActual);
if (!*ImageSizeActual) {
DEBUG ((EFI_D_ERROR, "Integer Overflow: ImgSizeActual=%u,"
" RecoveryDtboActual=%u\n", tempImgSize, RecoveryDtboActual));
return EFI_BAD_BUFFER_SIZE;
}
}
DEBUG ((EFI_D_VERBOSE, "Boot Image Header Info...\n"));
DEBUG ((EFI_D_VERBOSE, "Image Header version : 0x%x\n", HeaderVersion));
DEBUG ((EFI_D_VERBOSE, "Kernel Size 1 : 0x%x\n", KernelSize));
DEBUG ((EFI_D_VERBOSE, "Kernel Size 2 : 0x%x\n", SecondSize));
DEBUG ((EFI_D_VERBOSE, "Ramdisk Size : 0x%x\n", RamdiskSize));
DEBUG ((EFI_D_VERBOSE, "DTB Size : 0x%x\n", DtSize));
if (HeaderVersion >= BOOT_HEADER_VERSION_THREE) {
DEBUG ((EFI_D_VERBOSE, "Vendor Ramdisk Size : 0x%x\n",
VendorRamdiskSize));
}
return Status;
}
/**
Load image header from partition
@param[in] Pname Partition name.
@param[out] ImageHdrBuffer Supplies the address where a pointer to the image
buffer.
@param[out] ImageHdrSize The Pointer for image actual size.
@retval EFI_SUCCESS Load image from partition successfully.
@retval other Failed to Load image from partition.
**/
EFI_STATUS
LoadImageHeader (CHAR16 *Pname, VOID **ImageHdrBuffer, UINT32 *ImageHdrSize)
{
if (ImageHdrBuffer == NULL) {
return EFI_INVALID_PARAMETER;
}
if (!ADD_OF (BOOT_IMG_MAX_PAGE_SIZE, ALIGNMENT_MASK_4KB - 1)) {
DEBUG ((EFI_D_ERROR, "Integer Overflow: in ALIGNMENT_MASK_4KB addition\n"));
return EFI_BAD_BUFFER_SIZE;
}
*ImageHdrBuffer =
AllocatePages (ALIGN_PAGES (BOOT_IMG_MAX_PAGE_SIZE, ALIGNMENT_MASK_4KB));
if (!*ImageHdrBuffer) {
DEBUG ((EFI_D_ERROR, "Failed to allocate for Boot image Hdr\n"));
return EFI_BAD_BUFFER_SIZE;
}
*ImageHdrSize = BOOT_IMG_MAX_PAGE_SIZE;
return LoadImageFromPartition (*ImageHdrBuffer, ImageHdrSize, Pname);
}
/**
Load image from partition
@param[in] Pname Partition name.
@param[in] ImageBuffer Supplies the address where a pointer to the image
buffer.
@param[in] ImageSizeActual Actual size of the Image.
@param[in] PageSize The page size
@retval EFI_SUCCESS Load image from partition successfully.
@retval other Failed to Load image from partition.
**/
EFI_STATUS
LoadImage (CHAR16 *Pname, VOID **ImageBuffer,
UINT32 ImageSizeActual, UINT32 PageSize)
{
EFI_STATUS Status = EFI_SUCCESS;
UINT32 ImageSize = 0;
// Check for invalid ImageBuffer
if (ImageBuffer == NULL) {
return EFI_INVALID_PARAMETER;
} else {
*ImageBuffer = NULL;
}
ImageSize =
ADD_OF (ROUND_TO_PAGE (ImageSizeActual, (PageSize - 1)), PageSize);
if (!ImageSize) {
DEBUG ((EFI_D_ERROR, "Integer Overflow: ImgSize=%u\n", ImageSizeActual));
return EFI_BAD_BUFFER_SIZE;
}
if (!ADD_OF (ImageSize, ALIGNMENT_MASK_4KB - 1)) {
DEBUG ((EFI_D_ERROR, "Integer Overflow: in ALIGNMENT_MASK_4KB addition\n"));
return EFI_BAD_BUFFER_SIZE;
}
*ImageBuffer = AllocatePages (ALIGN_PAGES (ImageSize, ALIGNMENT_MASK_4KB));
if (!*ImageBuffer) {
DEBUG ((EFI_D_ERROR, "No resources available for ImageBuffer\n"));
return EFI_OUT_OF_RESOURCES;
}
BootStatsSetTimeStamp (BS_KERNEL_LOAD_START);
Status = LoadImageFromPartition (*ImageBuffer, &ImageSize, Pname);
BootStatsSetTimeStamp (BS_KERNEL_LOAD_DONE);
if (Status != EFI_SUCCESS) {
DEBUG ((EFI_D_ERROR, "Failed Kernel Size : 0x%x\n", ImageSize));
return Status;
}
return Status;
}
EFI_STATUS
GetImage (CONST BootInfo *Info,
VOID **ImageBuffer,
UINTN *ImageSize,
CHAR8 *ImageName)
{
if (Info == NULL || ImageBuffer == NULL || ImageSize == NULL ||
ImageName == NULL) {
DEBUG ((EFI_D_ERROR, "GetImage: invalid parameters\n"));
return EFI_INVALID_PARAMETER;
}
for (UINTN LoadedIndex = 0; LoadedIndex < Info->NumLoadedImages;
LoadedIndex++) {
if (!AsciiStrnCmp (Info->Images[LoadedIndex].Name, ImageName,
AsciiStrLen (ImageName))) {
*ImageBuffer = Info->Images[LoadedIndex].ImageBuffer;
*ImageSize = Info->Images[LoadedIndex].ImageSize;
return EFI_SUCCESS;
}
}
return EFI_NOT_FOUND;
}
/* Return Build variant */
#ifdef USER_BUILD_VARIANT
BOOLEAN TargetBuildVariantUser (VOID)
{
return TRUE;
}
#else
BOOLEAN TargetBuildVariantUser (VOID)
{
return FALSE;
}
#endif
#ifdef ENABLE_LE_VARIANT
BOOLEAN IsLEVariant (VOID)
{
return TRUE;
}
#else
BOOLEAN IsLEVariant (VOID)
{
return FALSE;
}
#endif
#ifdef BUILD_SYSTEM_ROOT_IMAGE
BOOLEAN IsBuildAsSystemRootImage (VOID)
{
return TRUE;
}
#else
BOOLEAN IsBuildAsSystemRootImage (VOID)
{
return FALSE;
}
#endif
#ifdef BUILD_USES_RECOVERY_AS_BOOT
BOOLEAN IsBuildUseRecoveryAsBoot (VOID)
{
return TRUE;
}
#else
BOOLEAN IsBuildUseRecoveryAsBoot (VOID)
{
return FALSE;
}
#endif
VOID
ResetBootDevImage (VOID)
{
BootDevImage = FALSE;
}
VOID
SetBootDevImage (VOID)
{
BootDevImage = TRUE;
}
BOOLEAN IsBootDevImage (VOID)
{
return BootDevImage;
}
#ifdef AB_RETRYCOUNT_DISABLE
BOOLEAN IsABRetryCountDisabled (VOID)
{
return TRUE;
}
#else
BOOLEAN IsABRetryCountDisabled (VOID)
{
return FALSE;
}
#endif
#if DYNAMIC_PARTITION_SUPPORT
BOOLEAN IsDynamicPartitionSupport (VOID)
{
return TRUE;
}
#else
BOOLEAN IsDynamicPartitionSupport (VOID)
{
return FALSE;
}
#endif
#if VIRTUAL_AB_OTA
BOOLEAN IsVirtualAbOtaSupported (VOID)
{
return TRUE;
}
#else
BOOLEAN IsVirtualAbOtaSupported (VOID)
{
return FALSE;
}
#endif
#if NAND_SQUASHFS_SUPPORT
BOOLEAN IsNANDSquashFsSupport (VOID)
{
return TRUE;
}
#else
BOOLEAN IsNANDSquashFsSupport (VOID)
{
return FALSE;
}
#endif
#if TARGET_BOARD_TYPE_AUTO
BOOLEAN IsEnableDisplayMenuFlagSupported (VOID)
{
return FALSE;
}
#else
BOOLEAN IsEnableDisplayMenuFlagSupported (VOID)
{
return TRUE;
}
#endif
#ifdef ENABLE_SYSTEMD_BOOTSLOT
BOOLEAN IsSystemdBootslotEnabled (VOID)
{
return TRUE;
}
#else
BOOLEAN IsSystemdBootslotEnabled (VOID)
{
return FALSE;
}
#endif