UefiPayloadPkg: Add a common SmmAccessDxe module
SmmAccessDxe module would consume EFI_SMRAM_HOB_DESCRIPTOR_BLOCK HOB to produce SMM access protocol gEfiSmmAccess2ProtocolGuid (open, close, lock, and GetCapabilities.) Signed-off-by: Guo Dong <guo.dong@intel.com> Cc: Ray Ni <ray.ni@intel.com> Cc: Maurice Ma <maurice.ma@intel.com> Cc: Benjamin You <benjamin.you@intel.com> Reviewed-by: Ray Ni <ray.ni@intel.com> Reviewed-by: Benjamin You <benjamin.you@intel.com>
This commit is contained in:
254
UefiPayloadPkg/SmmAccessDxe/SmmAccessDxe.c
Normal file
254
UefiPayloadPkg/SmmAccessDxe/SmmAccessDxe.c
Normal file
@@ -0,0 +1,254 @@
|
||||
/** @file
|
||||
This driver publishes the SMM Access 2 Protocol.
|
||||
|
||||
Copyright (c) 2021, Intel Corporation. All rights reserved.<BR>
|
||||
SPDX-License-Identifier: BSD-2-Clause-Patent
|
||||
|
||||
**/
|
||||
|
||||
#include "SmmAccessDxe.h"
|
||||
|
||||
SMM_ACCESS_PRIVATE_DATA mSmmAccess;
|
||||
|
||||
/**
|
||||
Update region state from SMRAM description
|
||||
|
||||
@param[in] OrLogic Indicate to use OR if true or AND if false.
|
||||
@param[in] Value The value to set to region state based on OrLogic.
|
||||
|
||||
**/
|
||||
VOID
|
||||
SyncRegionState2SmramDesc(
|
||||
IN BOOLEAN OrLogic,
|
||||
IN UINT64 Value
|
||||
)
|
||||
{
|
||||
UINT32 Index;
|
||||
|
||||
for (Index = 0; Index < mSmmAccess.NumberRegions; Index++) {
|
||||
if (OrLogic) {
|
||||
mSmmAccess.SmramDesc[Index].RegionState |= Value;
|
||||
} else {
|
||||
mSmmAccess.SmramDesc[Index].RegionState &= Value;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
This routine accepts a request to "open" a region of SMRAM. The
|
||||
region could be legacy ABSEG, HSEG, or TSEG near top of physical memory.
|
||||
The use of "open" means that the memory is visible from all boot-service
|
||||
and SMM agents.
|
||||
|
||||
@param This Pointer to the SMM Access Interface.
|
||||
|
||||
@retval EFI_SUCCESS The region was successfully opened.
|
||||
@retval EFI_DEVICE_ERROR The region could not be opened because locked by chipset.
|
||||
@retval EFI_INVALID_PARAMETER The descriptor index was out of bounds.
|
||||
|
||||
**/
|
||||
EFI_STATUS
|
||||
EFIAPI
|
||||
Open (
|
||||
IN EFI_SMM_ACCESS2_PROTOCOL *This
|
||||
)
|
||||
{
|
||||
if ((mSmmAccess.SmmRegionState & EFI_SMRAM_LOCKED) != 0) {
|
||||
//
|
||||
// Cannot open a "locked" region
|
||||
//
|
||||
DEBUG ((DEBUG_INFO, "Cannot open the locked SMRAM Region\n"));
|
||||
return EFI_DEVICE_ERROR;
|
||||
}
|
||||
|
||||
mSmmAccess.SmmRegionState &= ~(EFI_SMRAM_CLOSED | EFI_ALLOCATED);
|
||||
SyncRegionState2SmramDesc(FALSE, (UINT64)(UINTN)(~(EFI_SMRAM_CLOSED | EFI_ALLOCATED)));
|
||||
|
||||
mSmmAccess.SmmRegionState |= EFI_SMRAM_OPEN;
|
||||
SyncRegionState2SmramDesc(TRUE, EFI_SMRAM_OPEN);
|
||||
mSmmAccess.SmmAccess.OpenState = TRUE;
|
||||
|
||||
return EFI_SUCCESS;
|
||||
}
|
||||
|
||||
/**
|
||||
This routine accepts a request to "close" a region of SMRAM. The region
|
||||
could be legacy AB or TSEG near top of physical memory.
|
||||
The use of "close" means that the memory is only visible from SMM agents,
|
||||
not from BS or RT code.
|
||||
|
||||
@param This Pointer to the SMM Access Interface.
|
||||
|
||||
@retval EFI_SUCCESS The region was successfully closed.
|
||||
@retval EFI_DEVICE_ERROR The region could not be closed because locked by
|
||||
chipset.
|
||||
@retval EFI_INVALID_PARAMETER The descriptor index was out of bounds.
|
||||
|
||||
**/
|
||||
EFI_STATUS
|
||||
EFIAPI
|
||||
Close (
|
||||
IN EFI_SMM_ACCESS2_PROTOCOL *This
|
||||
)
|
||||
{
|
||||
if ((mSmmAccess.SmmRegionState & EFI_SMRAM_LOCKED) != 0) {
|
||||
//
|
||||
// Cannot close a "locked" region
|
||||
//
|
||||
DEBUG ((DEBUG_INFO, "Cannot close the locked SMRAM Region\n"));
|
||||
return EFI_DEVICE_ERROR;
|
||||
}
|
||||
|
||||
if ((mSmmAccess.SmmRegionState & EFI_SMRAM_CLOSED) != 0) {
|
||||
return EFI_DEVICE_ERROR;
|
||||
}
|
||||
|
||||
mSmmAccess.SmmRegionState &= ~EFI_SMRAM_OPEN;
|
||||
SyncRegionState2SmramDesc(FALSE, (UINT64)(UINTN)(~EFI_SMRAM_OPEN));
|
||||
|
||||
mSmmAccess.SmmRegionState |= (EFI_SMRAM_CLOSED | EFI_ALLOCATED);
|
||||
SyncRegionState2SmramDesc(TRUE, EFI_SMRAM_CLOSED | EFI_ALLOCATED);
|
||||
|
||||
mSmmAccess.SmmAccess.OpenState = FALSE;
|
||||
|
||||
return EFI_SUCCESS;
|
||||
}
|
||||
|
||||
/**
|
||||
This routine accepts a request to "lock" SMRAM. The
|
||||
region could be legacy AB or TSEG near top of physical memory.
|
||||
The use of "lock" means that the memory can no longer be opened
|
||||
to BS state.
|
||||
|
||||
@param This Pointer to the SMM Access Interface.
|
||||
|
||||
@retval EFI_SUCCESS The region was successfully locked.
|
||||
@retval EFI_DEVICE_ERROR The region could not be locked because at least
|
||||
one range is still open.
|
||||
@retval EFI_INVALID_PARAMETER The descriptor index was out of bounds.
|
||||
|
||||
**/
|
||||
EFI_STATUS
|
||||
EFIAPI
|
||||
Lock (
|
||||
IN EFI_SMM_ACCESS2_PROTOCOL *This
|
||||
)
|
||||
{
|
||||
if (mSmmAccess.SmmAccess.OpenState) {
|
||||
DEBUG ((DEBUG_INFO, "Cannot lock SMRAM when it is still open\n"));
|
||||
return EFI_DEVICE_ERROR;
|
||||
}
|
||||
|
||||
mSmmAccess.SmmRegionState |= EFI_SMRAM_LOCKED;
|
||||
SyncRegionState2SmramDesc(TRUE, EFI_SMRAM_LOCKED);
|
||||
mSmmAccess.SmmAccess.LockState = TRUE;
|
||||
return EFI_SUCCESS;
|
||||
}
|
||||
|
||||
/**
|
||||
This routine services a user request to discover the SMRAM
|
||||
capabilities of this platform. This will report the possible
|
||||
ranges that are possible for SMRAM access, based upon the
|
||||
memory controller capabilities.
|
||||
|
||||
@param This Pointer to the SMRAM Access Interface.
|
||||
@param SmramMapSize Pointer to the variable containing size of the
|
||||
buffer to contain the description information.
|
||||
@param SmramMap Buffer containing the data describing the Smram
|
||||
region descriptors.
|
||||
|
||||
@retval EFI_BUFFER_TOO_SMALL The user did not provide a sufficient buffer.
|
||||
@retval EFI_SUCCESS The user provided a sufficiently-sized buffer.
|
||||
|
||||
**/
|
||||
EFI_STATUS
|
||||
EFIAPI
|
||||
GetCapabilities (
|
||||
IN CONST EFI_SMM_ACCESS2_PROTOCOL *This,
|
||||
IN OUT UINTN *SmramMapSize,
|
||||
IN OUT EFI_SMRAM_DESCRIPTOR *SmramMap
|
||||
)
|
||||
{
|
||||
EFI_STATUS Status;
|
||||
UINTN NecessaryBufferSize;
|
||||
|
||||
NecessaryBufferSize = mSmmAccess.NumberRegions * sizeof(EFI_SMRAM_DESCRIPTOR);
|
||||
if (*SmramMapSize < NecessaryBufferSize) {
|
||||
Status = EFI_BUFFER_TOO_SMALL;
|
||||
} else {
|
||||
CopyMem(SmramMap, mSmmAccess.SmramDesc, NecessaryBufferSize);
|
||||
Status = EFI_SUCCESS;
|
||||
}
|
||||
|
||||
*SmramMapSize = NecessaryBufferSize;
|
||||
return Status;
|
||||
}
|
||||
|
||||
/**
|
||||
This function installs EFI_SMM_ACCESS_PROTOCOL.
|
||||
|
||||
@param ImageHandle Handle for the image of this driver
|
||||
@param SystemTable Pointer to the EFI System Table
|
||||
|
||||
@retval EFI_UNSUPPORTED There's no Intel ICH on this platform
|
||||
@return The status returned from InstallProtocolInterface().
|
||||
|
||||
**/
|
||||
EFI_STATUS
|
||||
EFIAPI
|
||||
SmmAccessEntryPoint (
|
||||
IN EFI_HANDLE ImageHandle,
|
||||
IN EFI_SYSTEM_TABLE *SystemTable
|
||||
)
|
||||
{
|
||||
EFI_STATUS Status;
|
||||
EFI_HOB_GUID_TYPE *GuidHob;
|
||||
UINT32 SmmRegionNum;
|
||||
EFI_SMRAM_HOB_DESCRIPTOR_BLOCK *SmramHob;
|
||||
UINT32 Index;
|
||||
|
||||
//
|
||||
// Get SMRAM info HOB
|
||||
//
|
||||
GuidHob = GetFirstGuidHob (&gEfiSmmSmramMemoryGuid);
|
||||
if (GuidHob == NULL) {
|
||||
DEBUG ((DEBUG_INFO, "SMRAM HOB NOT found\n"));
|
||||
return EFI_NOT_FOUND;
|
||||
}
|
||||
SmramHob = (EFI_SMRAM_HOB_DESCRIPTOR_BLOCK *) GET_GUID_HOB_DATA(GuidHob);
|
||||
SmmRegionNum = SmramHob->NumberOfSmmReservedRegions;
|
||||
mSmmAccess.SmramDesc = AllocateZeroPool (sizeof (EFI_SMRAM_DESCRIPTOR) * SmmRegionNum);
|
||||
if (mSmmAccess.SmramDesc == NULL) {
|
||||
return EFI_OUT_OF_RESOURCES;
|
||||
}
|
||||
CopyMem (mSmmAccess.SmramDesc, &SmramHob->Descriptor, sizeof (EFI_SMRAM_DESCRIPTOR) * SmmRegionNum);
|
||||
|
||||
DEBUG ((DEBUG_INFO, "NumberOfSmmReservedRegions = 0x%x\n", SmmRegionNum));
|
||||
for (Index = 0; Index < SmmRegionNum; Index++) {
|
||||
DEBUG ((DEBUG_INFO, "%d: base=0x%x, size = 0x%x, State=0x%x\n",Index,
|
||||
SmramHob->Descriptor[Index].PhysicalStart,
|
||||
SmramHob->Descriptor[Index].PhysicalSize,
|
||||
SmramHob->Descriptor[Index].RegionState));
|
||||
mSmmAccess.SmramDesc[Index].RegionState &= EFI_ALLOCATED;
|
||||
mSmmAccess.SmramDesc[Index].RegionState |= EFI_SMRAM_CLOSED | EFI_CACHEABLE;
|
||||
}
|
||||
|
||||
mSmmAccess.Signature = SMM_ACCESS_PRIVATE_DATA_SIGNATURE;
|
||||
mSmmAccess.NumberRegions = SmmRegionNum;
|
||||
mSmmAccess.SmmAccess.Open = Open;
|
||||
mSmmAccess.SmmAccess.Close = Close;
|
||||
mSmmAccess.SmmAccess.Lock = Lock;
|
||||
mSmmAccess.SmmAccess.GetCapabilities = GetCapabilities;
|
||||
mSmmAccess.SmmAccess.LockState = FALSE;
|
||||
mSmmAccess.SmmAccess.OpenState = FALSE;
|
||||
mSmmAccess.SmmRegionState = EFI_SMRAM_CLOSED;
|
||||
|
||||
Status = gBS->InstallMultipleProtocolInterfaces (
|
||||
&mSmmAccess.Handle,
|
||||
&gEfiSmmAccess2ProtocolGuid,
|
||||
&mSmmAccess.SmmAccess,
|
||||
NULL
|
||||
);
|
||||
|
||||
return Status;
|
||||
}
|
Reference in New Issue
Block a user