mirror of https://github.com/acidanthera/audk.git
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:
parent
8b76f23534
commit
87a34ca0cf
|
@ -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;
|
||||
}
|
|
@ -0,0 +1,37 @@
|
|||
/** @file
|
||||
The header file of SMM access DXE.
|
||||
|
||||
Copyright (c) 2014 - 2021, Intel Corporation. All rights reserved.<BR>
|
||||
SPDX-License-Identifier: BSD-2-Clause-Patent
|
||||
|
||||
**/
|
||||
|
||||
#ifndef SMM_ACCESS_DRIVER_H_
|
||||
#define SMM_ACCESS_DRIVER_H_
|
||||
|
||||
#include <PiDxe.h>
|
||||
#include <Protocol/SmmAccess2.h>
|
||||
#include <Library/HobLib.h>
|
||||
#include <Library/BaseLib.h>
|
||||
#include <Library/UefiBootServicesTableLib.h>
|
||||
#include <Library/DebugLib.h>
|
||||
#include <Library/MemoryAllocationLib.h>
|
||||
#include <Library/BaseMemoryLib.h>
|
||||
#include <Guid/SmramMemoryReserve.h>
|
||||
|
||||
|
||||
#define SMM_ACCESS_PRIVATE_DATA_SIGNATURE SIGNATURE_32 ('S', 'M', 'M', 'A')
|
||||
|
||||
typedef struct {
|
||||
UINTN Signature;
|
||||
EFI_HANDLE Handle;
|
||||
EFI_SMM_ACCESS2_PROTOCOL SmmAccess;
|
||||
//
|
||||
// Local Data for SMM Access interface goes here
|
||||
//
|
||||
UINT32 SmmRegionState;
|
||||
UINT32 NumberRegions;
|
||||
EFI_SMRAM_DESCRIPTOR *SmramDesc;
|
||||
} SMM_ACCESS_PRIVATE_DATA;
|
||||
|
||||
#endif
|
|
@ -0,0 +1,51 @@
|
|||
## @file
|
||||
# SMM Access 2 Protocol Dxe Driver
|
||||
#
|
||||
# This module produces the SMM Access 2 Protocol.
|
||||
#
|
||||
# Copyright (c) 2021, Intel Corporation. All rights reserved.<BR>
|
||||
#
|
||||
# SPDX-License-Identifier: BSD-2-Clause-Patent
|
||||
#
|
||||
##
|
||||
|
||||
[Defines]
|
||||
INF_VERSION = 0x00010005
|
||||
BASE_NAME = SmmAccessDxe
|
||||
FILE_GUID = 47579CF5-1E4F-4b41-99BB-A5C334846D3B
|
||||
MODULE_TYPE = DXE_DRIVER
|
||||
VERSION_STRING = 1.0
|
||||
ENTRY_POINT = SmmAccessEntryPoint
|
||||
|
||||
#
|
||||
# The following information is for reference only and not required by the build tools.
|
||||
#
|
||||
# VALID_ARCHITECTURES = IA32 X64
|
||||
#
|
||||
|
||||
[Sources]
|
||||
SmmAccessDxe.c
|
||||
SmmAccessDxe.h
|
||||
|
||||
[Packages]
|
||||
MdePkg/MdePkg.dec
|
||||
MdeModulePkg/MdeModulePkg.dec
|
||||
UefiPayloadPkg/UefiPayloadPkg.dec
|
||||
|
||||
[LibraryClasses]
|
||||
UefiDriverEntryPoint
|
||||
UefiBootServicesTableLib
|
||||
DebugLib
|
||||
BaseLib
|
||||
BaseMemoryLib
|
||||
MemoryAllocationLib
|
||||
HobLib
|
||||
|
||||
[Guids]
|
||||
gEfiSmmSmramMemoryGuid
|
||||
|
||||
[Protocols]
|
||||
gEfiSmmAccess2ProtocolGuid ## PRODUCES
|
||||
|
||||
[Depex]
|
||||
TRUE
|
Loading…
Reference in New Issue