/*++ | |
Copyright (c) 2006 - 2010, Intel Corporation. All rights reserved.<BR> | |
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. | |
Module Name: | |
EfiLoader.c | |
Abstract: | |
Revision History: | |
--*/ | |
#include "EfiLdr.h" | |
#include "Support.h" | |
#include "Debug.h" | |
#include "PeLoader.h" | |
#include "LzmaDecompress.h" | |
VOID | |
SystemHang( | |
VOID | |
) | |
{ | |
CHAR8 PrintBuffer[256]; | |
AsciiSPrint (PrintBuffer, 256, "## FATEL ERROR ##: Fail to load DUET images! System hang!\n"); | |
CpuDeadLoop(); | |
} | |
VOID | |
EfiLoader ( | |
UINT32 BiosMemoryMapBaseAddress | |
) | |
{ | |
BIOS_MEMORY_MAP *BiosMemoryMap; | |
EFILDR_IMAGE *EFILDRImage; | |
EFI_MEMORY_DESCRIPTOR EfiMemoryDescriptor[EFI_MAX_MEMORY_DESCRIPTORS]; | |
EFI_STATUS Status; | |
UINTN NumberOfMemoryMapEntries; | |
UINT32 DestinationSize; | |
UINT32 ScratchSize; | |
UINTN BfvPageNumber; | |
UINTN BfvBase; | |
EFI_MAIN_ENTRYPOINT EfiMainEntrypoint; | |
CHAR8 PrintBuffer[256]; | |
EFILDRHANDOFF Handoff; | |
ClearScreen(); | |
PrintHeader ('A'); | |
AsciiSPrint (PrintBuffer, 256, "Enter DUET Loader...\n"); | |
PrintString (PrintBuffer); | |
AsciiSPrint (PrintBuffer, 256, "BiosMemoryMapBaseAddress = 0x%x\n", BiosMemoryMapBaseAddress); | |
PrintString (PrintBuffer); | |
// | |
// Add all EfiConventionalMemory descriptors to the table. If there are partial pages, then | |
// round the start address up to the next page, and round the length down to a page boundry. | |
// | |
BiosMemoryMap = (BIOS_MEMORY_MAP *)(UINTN)(BiosMemoryMapBaseAddress); | |
NumberOfMemoryMapEntries = 0; | |
GenMemoryMap (&NumberOfMemoryMapEntries, EfiMemoryDescriptor, BiosMemoryMap); | |
AsciiSPrint (PrintBuffer, 256, "Get %d entries of memory map!\n", NumberOfMemoryMapEntries); | |
PrintString (PrintBuffer); | |
// | |
// Get information on where the image is in memory | |
// | |
//EFILDRHeader = (EFILDR_HEADER *)(UINTN)(EFILDR_HEADER_ADDRESS); | |
EFILDRImage = (EFILDR_IMAGE *)(UINTN)(EFILDR_HEADER_ADDRESS + sizeof(EFILDR_HEADER)); | |
// | |
// Point to the 4th image (Bfv) | |
// | |
EFILDRImage += 3; | |
// | |
// Decompress the image | |
// | |
AsciiSPrint (PrintBuffer, 256, "Decompress BFV image, Image Address=0x%x Offset=0x%x\n", | |
(UINTN)(EFILDR_HEADER_ADDRESS + EFILDRImage->Offset), | |
EFILDRImage->Offset); | |
PrintString (PrintBuffer); | |
Status = LzmaUefiDecompressGetInfo ( | |
(VOID *)(UINTN)(EFILDR_HEADER_ADDRESS + EFILDRImage->Offset), | |
EFILDRImage->Length, | |
&DestinationSize, | |
&ScratchSize | |
); | |
if (EFI_ERROR (Status)) { | |
AsciiSPrint (PrintBuffer, 256, "Fail to get decompress information for BFV!\n"); | |
PrintString (PrintBuffer); | |
SystemHang(); | |
} | |
AsciiSPrint (PrintBuffer, 256, "BFV decompress: DestinationSize=0x%X, ScratchSize=0x%X!\n", | |
DestinationSize, ScratchSize); | |
PrintString (PrintBuffer); | |
Status = LzmaUefiDecompress ( | |
(VOID *)(UINTN)(EFILDR_HEADER_ADDRESS + EFILDRImage->Offset), | |
EFILDRImage->Length, | |
(VOID *)(UINTN)EFI_DECOMPRESSED_BUFFER_ADDRESS, | |
(VOID *)(UINTN)((EFI_DECOMPRESSED_BUFFER_ADDRESS + DestinationSize + 0x1000) & 0xfffff000) | |
); | |
if (EFI_ERROR (Status)) { | |
AsciiSPrint (PrintBuffer, 256, "Fail to decompress BFV!\n"); | |
PrintString (PrintBuffer); | |
SystemHang(); | |
} | |
BfvPageNumber = EFI_SIZE_TO_PAGES (DestinationSize); | |
BfvBase = (UINTN) FindSpace (BfvPageNumber, &NumberOfMemoryMapEntries, EfiMemoryDescriptor, EfiRuntimeServicesData, EFI_MEMORY_WB); | |
if (BfvBase == 0) { | |
SystemHang(); | |
} | |
ZeroMem ((VOID *)(UINTN)BfvBase, BfvPageNumber * EFI_PAGE_SIZE); | |
CopyMem ((VOID *)(UINTN)BfvBase, (VOID *)(UINTN)EFI_DECOMPRESSED_BUFFER_ADDRESS, DestinationSize); | |
PrintHeader ('B'); | |
// | |
// Point to the 2nd image (DxeIpl) | |
// | |
EFILDRImage -= 2; | |
// | |
// Decompress the image | |
// | |
AsciiSPrint (PrintBuffer, 256, "Decompress DxeIpl image, Image Address=0x%x Offset=0x%x\n", | |
(UINTN)(EFILDR_HEADER_ADDRESS + EFILDRImage->Offset), | |
EFILDRImage->Offset); | |
PrintString (PrintBuffer); | |
Status = LzmaUefiDecompressGetInfo ( | |
(VOID *)(UINTN)(EFILDR_HEADER_ADDRESS + EFILDRImage->Offset), | |
EFILDRImage->Length, | |
&DestinationSize, | |
&ScratchSize | |
); | |
if (EFI_ERROR (Status)) { | |
AsciiSPrint (PrintBuffer, 256, "Fail to get decompress information for DxeIpl!\n"); | |
PrintString (PrintBuffer); | |
SystemHang(); | |
} | |
Status = LzmaUefiDecompress ( | |
(VOID *)(UINTN)(EFILDR_HEADER_ADDRESS + EFILDRImage->Offset), | |
EFILDRImage->Length, | |
(VOID *)(UINTN)EFI_DECOMPRESSED_BUFFER_ADDRESS, | |
(VOID *)(UINTN)((EFI_DECOMPRESSED_BUFFER_ADDRESS + DestinationSize + 0x1000) & 0xfffff000) | |
); | |
if (EFI_ERROR (Status)) { | |
AsciiSPrint (PrintBuffer, 256, "Fail to decompress DxeIpl image\n"); | |
PrintString (PrintBuffer); | |
SystemHang(); | |
} | |
AsciiSPrint (PrintBuffer, 256, "Start load DxeIpl PE image\n"); | |
PrintString (PrintBuffer); | |
// | |
// Load and relocate the EFI PE/COFF Firmware Image | |
// | |
Status = EfiLdrPeCoffLoadPeImage ( | |
(VOID *)(UINTN)(EFI_DECOMPRESSED_BUFFER_ADDRESS), | |
&DxeIplImage, | |
&NumberOfMemoryMapEntries, | |
EfiMemoryDescriptor | |
); | |
if (EFI_ERROR (Status)) { | |
AsciiSPrint (PrintBuffer, 256, "Fail to load and relocate DxeIpl PE image!\n"); | |
PrintString (PrintBuffer); | |
SystemHang(); | |
} | |
AsciiSPrint (PrintBuffer, 256, "DxeIpl PE image is successed loaded at 0x%x, entry=0x%x\n", | |
(UINTN)DxeIplImage.ImageBasePage, (UINTN)DxeIplImage.EntryPoint); | |
PrintString (PrintBuffer); | |
// PrintString("Image.NoPages = "); | |
// PrintValue(Image.NoPages); | |
// PrintString("\n"); | |
PrintHeader ('C'); | |
// | |
// Point to the 3rd image (DxeMain) | |
// | |
EFILDRImage++; | |
// | |
// Decompress the image | |
// | |
AsciiSPrint (PrintBuffer, 256, "Decompress DXEMain FV image, Image Address=0x%x! Offset=0x%x\n", | |
(UINTN)(EFILDR_HEADER_ADDRESS + EFILDRImage->Offset), | |
EFILDRImage->Offset); | |
PrintString (PrintBuffer); | |
Status = LzmaUefiDecompressGetInfo ( | |
(VOID *)(UINTN)(EFILDR_HEADER_ADDRESS + EFILDRImage->Offset), | |
EFILDRImage->Length, | |
&DestinationSize, | |
&ScratchSize | |
); | |
if (EFI_ERROR (Status)) { | |
AsciiSPrint (PrintBuffer, 256, "Fail to get decompress information for DXEMain FV image!\n"); | |
PrintString (PrintBuffer); | |
SystemHang(); | |
} | |
Status = LzmaUefiDecompress ( | |
(VOID *)(UINTN)(EFILDR_HEADER_ADDRESS + EFILDRImage->Offset), | |
EFILDRImage->Length, | |
(VOID *)(UINTN)EFI_DECOMPRESSED_BUFFER_ADDRESS, | |
(VOID *)(UINTN)((EFI_DECOMPRESSED_BUFFER_ADDRESS + DestinationSize + 0x1000) & 0xfffff000) | |
); | |
if (EFI_ERROR (Status)) { | |
SystemHang(); | |
} | |
// | |
// Load and relocate the EFI PE/COFF Firmware Image | |
// | |
Status = EfiLdrPeCoffLoadPeImage ( | |
(VOID *)(UINTN)(EFI_DECOMPRESSED_BUFFER_ADDRESS), | |
&DxeCoreImage, | |
&NumberOfMemoryMapEntries, | |
EfiMemoryDescriptor | |
); | |
if (EFI_ERROR (Status)) { | |
SystemHang(); | |
} | |
AsciiSPrint (PrintBuffer, 256, "DxeCore PE image is successed loaded at 0x%x, entry=0x%x\n", | |
(UINTN)DxeCoreImage.ImageBasePage, (UINTN)DxeCoreImage.EntryPoint); | |
PrintString (PrintBuffer); | |
PrintHeader ('E'); | |
// | |
// Display the table of memory descriptors. | |
// | |
// PrintString("\nEFI Memory Descriptors\n"); | |
/* | |
{ | |
UINTN Index; | |
for (Index = 0; Index < NumberOfMemoryMapEntries; Index++) { | |
PrintString("Type = "); | |
PrintValue(EfiMemoryDescriptor[Index].Type); | |
PrintString(" Start = "); | |
PrintValue((UINT32)(EfiMemoryDescriptor[Index].PhysicalStart)); | |
PrintString(" NumberOfPages = "); | |
PrintValue((UINT32)(EfiMemoryDescriptor[Index].NumberOfPages)); | |
PrintString("\n"); | |
} | |
} | |
*/ | |
// | |
// Jump to EFI Firmware | |
// | |
if (DxeIplImage.EntryPoint != NULL) { | |
Handoff.MemDescCount = NumberOfMemoryMapEntries; | |
Handoff.MemDesc = EfiMemoryDescriptor; | |
Handoff.BfvBase = (VOID *)(UINTN)BfvBase; | |
Handoff.BfvSize = BfvPageNumber * EFI_PAGE_SIZE; | |
Handoff.DxeIplImageBase = (VOID *)(UINTN)DxeIplImage.ImageBasePage; | |
Handoff.DxeIplImageSize = DxeIplImage.NoPages * EFI_PAGE_SIZE; | |
Handoff.DxeCoreImageBase = (VOID *)(UINTN)DxeCoreImage.ImageBasePage; | |
Handoff.DxeCoreImageSize = DxeCoreImage.NoPages * EFI_PAGE_SIZE; | |
Handoff.DxeCoreEntryPoint = (VOID *)(UINTN)DxeCoreImage.EntryPoint; | |
AsciiSPrint (PrintBuffer, 256, "Transfer to DxeIpl ...Address=0x%x\n", (UINTN)DxeIplImage.EntryPoint); | |
PrintString (PrintBuffer); | |
EfiMainEntrypoint = (EFI_MAIN_ENTRYPOINT)(UINTN)DxeIplImage.EntryPoint; | |
EfiMainEntrypoint (&Handoff); | |
} | |
PrintHeader ('F'); | |
// | |
// There was a problem loading the image, so HALT the system. | |
// | |
SystemHang(); | |
} | |
EFI_STATUS | |
EFIAPI | |
_ModuleEntryPoint ( | |
UINT32 BiosMemoryMapBaseAddress | |
) | |
{ | |
EfiLoader(BiosMemoryMapBaseAddress); | |
return EFI_SUCCESS; | |
} | |