blob: a95e24958bd4eaa02051ef8a9a0a46bbfffbd8d1 [file] [log] [blame]
/** @file
*
* Copyright (c) 2014, ARM Limited. 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 <PiDxe.h>
#include <Library/DebugLib.h>
#include <Library/FdtLoadLib.h>
#include <Library/MemoryAllocationLib.h>
#include <Library/UefiBootServicesTableLib.h>
#include <Protocol/DevicePath.h>
#include <Protocol/FirmwareVolume2.h>
#include <Protocol/SimpleFileSystem.h>
#include <Guid/Fdt.h>
#include <Guid/FileInfo.h>
#include <libfdt.h>
//
// Device path for SemiHosting
//
STATIC CONST struct {
VENDOR_DEVICE_PATH Guid;
EFI_DEVICE_PATH_PROTOCOL End;
} mSemihostingDevicePath = {
{
{ HARDWARE_DEVICE_PATH, HW_VENDOR_DP, { sizeof (VENDOR_DEVICE_PATH), 0 } },
{ 0xC5B9C74A, 0x6D72, 0x4719, { 0x99, 0xAB, 0xC5, 0x9F, 0x19, 0x90, 0x91, 0xEB } }
},
{ END_DEVICE_PATH_TYPE, END_ENTIRE_DEVICE_PATH_SUBTYPE, { sizeof (EFI_DEVICE_PATH_PROTOCOL), 0 } }
};
/**
This function declares the passed FDT into the UEFI Configuration Table
@param FdtBlob Base address of the Fdt Blob in System Memory
@param FdtSize Size of the Fdt Blob in System Memory
@return EFI_SUCCESS Fdt Blob was successfully installed into the configuration table
@return !EFI_SUCCESS Error returned by BS.InstallConfigurationTable()
**/
STATIC
EFI_STATUS
InstallFdtIntoConfigurationTable (
IN VOID* FdtBlob,
IN UINTN FdtSize
)
{
EFI_STATUS Status;
// Check the FDT header is valid. We only make this check in DEBUG mode in case the FDT header change on
// production device and this ASSERT() becomes not valid.
ASSERT (fdt_check_header (FdtBlob) == 0);
// Ensure the Size of the Device Tree is smaller than the size of the read file
ASSERT ((UINTN)fdt_totalsize (FdtBlob) <= FdtSize);
// Install the FDT into the Configuration Table
Status = gBS->InstallConfigurationTable (&gFdtTableGuid, FdtBlob);
return Status;
}
/**
Load and Install FDT from Semihosting
@param Filename Name of the file to load from semihosting
@return EFI_SUCCESS Fdt Blob was successfully installed into the configuration table
from semihosting
@return EFI_NOT_FOUND Fail to locate the file in semihosting
@return EFI_OUT_OF_RESOURCES Fail to allocate memory to contain the blob
**/
EFI_STATUS
InstallFdtFromSemihosting (
IN CONST CHAR16* FileName
)
{
EFI_STATUS Status;
EFI_DEVICE_PATH* Remaining;
EFI_HANDLE Handle;
EFI_SIMPLE_FILE_SYSTEM_PROTOCOL *SemihostingFs;
EFI_FILE_PROTOCOL *Fs;
EFI_FILE_PROTOCOL *File;
EFI_PHYSICAL_ADDRESS FdtBase;
EFI_FILE_INFO *FileInfo;
UINTN FdtSize;
UINTN FileInfoSize;
// Ensure the Semihosting driver is initialized
Remaining = (EFI_DEVICE_PATH*)&mSemihostingDevicePath;
// The LocateDevicePath() function locates all devices on DevicePath that support Protocol and returns
// the handle to the device that is closest to DevicePath. On output, the device path pointer is modified
// to point to the remaining part of the device path
Status = gBS->LocateDevicePath (&gEfiDevicePathProtocolGuid, &Remaining, &Handle);
if (EFI_ERROR (Status)) {
ASSERT_EFI_ERROR (Status);
return Status;
}
// Recursive = FALSE: We do not want to start the whole device tree
Status = gBS->ConnectController (Handle, NULL, Remaining, FALSE);
if (EFI_ERROR (Status)) {
return Status;
}
// Locate the FileSystem
Status = gBS->HandleProtocol (Handle, &gEfiSimpleFileSystemProtocolGuid, (VOID **)&SemihostingFs);
if (EFI_ERROR (Status)) {
ASSERT_EFI_ERROR (Status);
return Status;
}
// Try to Open the volume and get root directory
Status = SemihostingFs->OpenVolume (SemihostingFs, &Fs);
if (EFI_ERROR (Status)) {
DEBUG ((EFI_D_WARN, "Warning: Fail to open semihosting filesystem that should contain FDT file.\n"));
return Status;
}
File = NULL;
Status = Fs->Open (Fs, &File, (CHAR16*)FileName, EFI_FILE_MODE_READ, 0);
if (EFI_ERROR (Status)) {
DEBUG ((EFI_D_WARN, "Warning: Fail to load FDT file '%s'.\n", FileName));
Fs->Close (Fs);
return Status;
}
FileInfoSize = 0;
File->GetInfo (File, &gEfiFileInfoGuid, &FileInfoSize, NULL);
FileInfo = AllocatePool (FileInfoSize);
if (FileInfo == NULL) {
Status = EFI_OUT_OF_RESOURCES;
goto CLOSE_FILES;
}
Status = File->GetInfo (File, &gEfiFileInfoGuid, &FileInfoSize, FileInfo);
if (EFI_ERROR (Status)) {
FreePool (FileInfo);
goto CLOSE_FILES;
}
// Get the file size
FdtSize = FileInfo->FileSize;
FreePool (FileInfo);
// The FDT blob is attached to the Configuration Table. It is recommended to load it as Runtime Service Data
// to prevent the kernel to overwrite its data
Status = gBS->AllocatePages (AllocateAnyPages, EfiRuntimeServicesData, EFI_SIZE_TO_PAGES (FdtSize), &FdtBase);
if (!EFI_ERROR (Status)) {
Status = File->Read (File, &FdtSize, (VOID*)(UINTN)(FdtBase));
if (EFI_ERROR (Status)) {
gBS->FreePages (FdtBase, EFI_SIZE_TO_PAGES (FdtSize));
} else {
// Install the FDT as part of the UEFI Configuration Table
Status = InstallFdtIntoConfigurationTable ((VOID*)(UINTN)FdtBase, FdtSize);
if (EFI_ERROR (Status)) {
gBS->FreePages (FdtBase, EFI_SIZE_TO_PAGES (FdtSize));
}
}
}
CLOSE_FILES:
File->Close (File);
Fs->Close (Fs);
return Status;
}
/**
Load and Install FDT from Firmware Volume
@param Filename Guid of the FDT blob to load from firmware volume
@return EFI_SUCCESS Fdt Blob was successfully installed into the configuration table
from firmware volume
@return EFI_NOT_FOUND Fail to locate the file in firmware volume
@return EFI_OUT_OF_RESOURCES Fail to allocate memory to contain the blob
**/
EFI_STATUS
InstallFdtFromFv (
IN CONST EFI_GUID *FileName
)
{
EFI_STATUS Status;
EFI_HANDLE *HandleBuffer;
UINTN NumberOfHandles;
UINT32 FvStatus;
UINTN Index;
EFI_FIRMWARE_VOLUME2_PROTOCOL *FvInstance;
INTN SectionInstance;
UINTN FdtSize;
VOID* FdtBlob;
EFI_PHYSICAL_ADDRESS FdtBase;
FvStatus = 0;
SectionInstance = 0;
// Locate all the Firmware Volume protocols.
Status = gBS->LocateHandleBuffer (
ByProtocol,
&gEfiFirmwareVolume2ProtocolGuid,
NULL,
&NumberOfHandles,
&HandleBuffer
);
if (EFI_ERROR (Status)) {
return Status;
}
// Looking for FV that contains the FDT blob
for (Index = 0; Index < NumberOfHandles; Index++) {
//
// Get the protocol on this handle
// This should not fail because of LocateHandleBuffer
//
Status = gBS->HandleProtocol (
HandleBuffer[Index],
&gEfiFirmwareVolume2ProtocolGuid,
(VOID**) &FvInstance
);
if (EFI_ERROR (Status)) {
goto FREE_HANDLE_BUFFER;
}
while (Status == EFI_SUCCESS) {
// FdtBlob must be allocated by ReadSection
FdtBlob = NULL;
// See if it contains the FDT file
Status = FvInstance->ReadSection (
FvInstance,
FileName,
EFI_SECTION_RAW,
SectionInstance,
&FdtBlob,
&FdtSize,
&FvStatus
);
if (!EFI_ERROR (Status)) {
// When the FDT blob is attached to the Configuration Table it is recommended to load it as Runtime Service Data
// to prevent the kernel to overwrite its data
Status = gBS->AllocatePages (AllocateAnyPages, EfiRuntimeServicesData, EFI_SIZE_TO_PAGES (FdtSize), &FdtBase);
if (EFI_ERROR (Status)) {
goto FREE_HANDLE_BUFFER;
}
// Copy the FDT to the Runtime memory
gBS->CopyMem ((VOID*)(UINTN)FdtBase, FdtBlob, FdtSize);
// Free the buffer allocated by FvInstance->ReadSection()
gBS->FreePool (FdtBlob);
// Install the FDT as part of the UEFI Configuration Table
Status = InstallFdtIntoConfigurationTable ((VOID*)(UINTN)FdtBase, FdtSize);
if (EFI_ERROR (Status)) {
gBS->FreePages (FdtBase, EFI_SIZE_TO_PAGES (FdtSize));
}
break;
}
}
}
FREE_HANDLE_BUFFER:
// Free any allocated buffers
gBS->FreePool (HandleBuffer);
return Status;
}