2011-07-12 22:35:46 +02:00
/** @file
Capsule update PEIM for UEFI2 .0
2019-06-10 03:38:18 +02:00
Copyright ( c ) 2006 - 2019 , Intel Corporation . All rights reserved . < BR >
2017-02-26 18:43:04 +01:00
Copyright ( c ) 2017 , AMD Incorporated . All rights reserved . < BR >
2011-07-12 22:35:46 +02:00
2019-04-04 01:05:13 +02:00
SPDX - License - Identifier : BSD - 2 - Clause - Patent
2011-07-12 22:35:46 +02:00
* */
# include "Capsule.h"
2019-06-10 03:52:22 +02:00
# define DEFAULT_SG_LIST_HEADS (20)
2011-09-05 11:53:37 +02:00
# ifdef MDE_CPU_IA32
2011-09-02 13:34:35 +02:00
//
// Global Descriptor Table (GDT)
//
GLOBAL_REMOVE_IF_UNREFERENCED IA32_SEGMENT_DESCRIPTOR mGdtEntries [ ] = {
/* selector { Global Segment Descriptor } */
/* 0x00 */ {
{ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 }
} , // null descriptor
/* 0x08 */ {
{ 0xffff , 0 , 0 , 0x3 , 1 , 0 , 1 , 0xf , 0 , 0 , 1 , 1 , 0 }
} , // linear data segment descriptor
/* 0x10 */ {
{ 0xffff , 0 , 0 , 0xf , 1 , 0 , 1 , 0xf , 0 , 0 , 1 , 1 , 0 }
} , // linear code segment descriptor
/* 0x18 */ {
{ 0xffff , 0 , 0 , 0x3 , 1 , 0 , 1 , 0xf , 0 , 0 , 1 , 1 , 0 }
} , // system data segment descriptor
/* 0x20 */ {
{ 0xffff , 0 , 0 , 0xb , 1 , 0 , 1 , 0xf , 0 , 0 , 1 , 1 , 0 }
} , // system code segment descriptor
/* 0x28 */ {
{ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 }
} , // spare segment descriptor
/* 0x30 */ {
{ 0xffff , 0 , 0 , 0x3 , 1 , 0 , 1 , 0xf , 0 , 0 , 1 , 1 , 0 }
} , // system data segment descriptor
/* 0x38 */ {
{ 0xffff , 0 , 0 , 0xb , 1 , 0 , 1 , 0xf , 0 , 1 , 0 , 1 , 0 }
} , // system code segment descriptor
/* 0x40 */ {
{ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 }
} , // spare segment descriptor
} ;
//
// IA32 Gdt register
//
GLOBAL_REMOVE_IF_UNREFERENCED CONST IA32_DESCRIPTOR mGdt = {
sizeof ( mGdtEntries ) - 1 ,
( UINTN ) mGdtEntries
} ;
2017-02-26 18:43:04 +01:00
2011-07-12 22:35:46 +02:00
/**
2015-07-27 05:04:41 +02:00
The function will check if 1 G page is supported .
@ retval TRUE 1 G page is supported .
@ retval FALSE 1 G page is not supported .
2011-07-12 22:35:46 +02:00
* */
2015-07-27 05:04:41 +02:00
BOOLEAN
IsPage1GSupport (
2011-09-02 13:34:35 +02:00
VOID
2011-07-12 22:35:46 +02:00
)
{
2011-09-13 07:48:57 +02:00
UINT32 RegEax ;
UINT32 RegEdx ;
BOOLEAN Page1GSupport ;
Page1GSupport = FALSE ;
2011-09-21 05:45:35 +02:00
if ( PcdGetBool ( PcdUse1GPageTable ) ) {
AsmCpuid ( 0x80000000 , & RegEax , NULL , NULL , NULL ) ;
if ( RegEax > = 0x80000001 ) {
AsmCpuid ( 0x80000001 , NULL , NULL , NULL , & RegEdx ) ;
if ( ( RegEdx & BIT26 ) ! = 0 ) {
Page1GSupport = TRUE ;
}
2011-09-13 07:48:57 +02:00
}
}
2011-09-02 13:34:35 +02:00
2015-07-27 05:04:41 +02:00
return Page1GSupport ;
}
/**
Calculate the total size of page table .
@ param [ in ] Page1GSupport 1 G page support or not .
@ return The size of page table .
* */
UINTN
CalculatePageTableSize (
IN BOOLEAN Page1GSupport
)
{
UINTN ExtraPageTablePages ;
UINTN TotalPagesNum ;
UINT8 PhysicalAddressBits ;
UINT32 NumberOfPml4EntriesNeeded ;
UINT32 NumberOfPdpEntriesNeeded ;
2011-09-02 13:34:35 +02:00
2011-07-12 22:35:46 +02:00
//
2015-07-27 05:04:41 +02:00
// Create 4G page table by default,
// and let PF handler to handle > 4G request.
2011-07-12 22:35:46 +02:00
//
2015-07-27 05:04:41 +02:00
PhysicalAddressBits = 32 ;
ExtraPageTablePages = EXTRA_PAGE_TABLE_PAGES ;
2011-09-02 13:34:35 +02:00
//
// Calculate the table entries needed.
//
if ( PhysicalAddressBits < = 39 ) {
NumberOfPml4EntriesNeeded = 1 ;
2011-09-13 07:48:57 +02:00
NumberOfPdpEntriesNeeded = ( UINT32 ) LShiftU64 ( 1 , ( PhysicalAddressBits - 30 ) ) ;
2011-09-02 13:34:35 +02:00
} else {
2011-09-13 07:48:57 +02:00
NumberOfPml4EntriesNeeded = ( UINT32 ) LShiftU64 ( 1 , ( PhysicalAddressBits - 39 ) ) ;
2011-09-02 13:34:35 +02:00
NumberOfPdpEntriesNeeded = 512 ;
2011-07-12 22:35:46 +02:00
}
2011-09-13 07:48:57 +02:00
if ( ! Page1GSupport ) {
TotalPagesNum = ( NumberOfPdpEntriesNeeded + 1 ) * NumberOfPml4EntriesNeeded + 1 ;
} else {
TotalPagesNum = NumberOfPml4EntriesNeeded + 1 ;
}
2021-12-05 23:54:02 +01:00
2015-07-27 05:04:41 +02:00
TotalPagesNum + = ExtraPageTablePages ;
2011-09-02 13:34:35 +02:00
return EFI_PAGES_TO_SIZE ( TotalPagesNum ) ;
2011-07-12 22:35:46 +02:00
}
/**
2011-09-02 13:34:35 +02:00
Allocates and fills in the Page Directory and Page Table Entries to
2015-07-27 05:04:41 +02:00
establish a 4 G page table .
2011-07-12 22:35:46 +02:00
2015-07-27 05:04:41 +02:00
@ param [ in ] PageTablesAddress The base address of page table .
@ param [ in ] Page1GSupport 1 G page support or not .
2011-07-12 22:35:46 +02:00
* */
2011-09-02 13:34:35 +02:00
VOID
2015-07-27 05:04:41 +02:00
Create4GPageTables (
IN EFI_PHYSICAL_ADDRESS PageTablesAddress ,
IN BOOLEAN Page1GSupport
2011-07-12 22:35:46 +02:00
)
2018-06-27 15:08:52 +02:00
{
2011-09-02 13:34:35 +02:00
UINT8 PhysicalAddressBits ;
EFI_PHYSICAL_ADDRESS PageAddress ;
UINTN IndexOfPml4Entries ;
UINTN IndexOfPdpEntries ;
UINTN IndexOfPageDirectoryEntries ;
UINT32 NumberOfPml4EntriesNeeded ;
UINT32 NumberOfPdpEntriesNeeded ;
PAGE_MAP_AND_DIRECTORY_POINTER * PageMapLevel4Entry ;
PAGE_MAP_AND_DIRECTORY_POINTER * PageMap ;
PAGE_MAP_AND_DIRECTORY_POINTER * PageDirectoryPointerEntry ;
PAGE_TABLE_ENTRY * PageDirectoryEntry ;
UINTN BigPageAddress ;
2011-09-13 07:48:57 +02:00
PAGE_TABLE_1G_ENTRY * PageDirectory1GEntry ;
2017-02-26 18:43:04 +01:00
UINT64 AddressEncMask ;
//
// Make sure AddressEncMask is contained to smallest supported address field.
//
AddressEncMask = PcdGet64 ( PcdPteMemoryEncryptionAddressOrMask ) & PAGING_1G_ADDRESS_MASK_64 ;
2011-09-13 07:48:57 +02:00
2011-09-02 13:34:35 +02:00
//
2015-07-27 05:04:41 +02:00
// Create 4G page table by default,
// and let PF handler to handle > 4G request.
2011-09-02 13:34:35 +02:00
//
2015-07-27 05:04:41 +02:00
PhysicalAddressBits = 32 ;
2011-07-12 22:35:46 +02:00
//
2011-09-02 13:34:35 +02:00
// Calculate the table entries needed.
//
if ( PhysicalAddressBits < = 39 ) {
NumberOfPml4EntriesNeeded = 1 ;
2011-09-13 07:48:57 +02:00
NumberOfPdpEntriesNeeded = ( UINT32 ) LShiftU64 ( 1 , ( PhysicalAddressBits - 30 ) ) ;
2011-09-02 13:34:35 +02:00
} else {
2011-09-13 07:48:57 +02:00
NumberOfPml4EntriesNeeded = ( UINT32 ) LShiftU64 ( 1 , ( PhysicalAddressBits - 39 ) ) ;
2011-09-02 13:34:35 +02:00
NumberOfPdpEntriesNeeded = 512 ;
}
//
2018-06-27 15:08:52 +02:00
// Pre-allocate big pages to avoid later allocations.
2011-09-02 13:34:35 +02:00
//
BigPageAddress = ( UINTN ) PageTablesAddress ;
//
// By architecture only one PageMapLevel4 exists - so lets allocate storage for it.
//
PageMap = ( VOID * ) BigPageAddress ;
2011-09-13 07:48:57 +02:00
BigPageAddress + = SIZE_4KB ;
2011-09-02 13:34:35 +02:00
PageMapLevel4Entry = PageMap ;
PageAddress = 0 ;
for ( IndexOfPml4Entries = 0 ; IndexOfPml4Entries < NumberOfPml4EntriesNeeded ; IndexOfPml4Entries + + , PageMapLevel4Entry + + ) {
2011-07-12 22:35:46 +02:00
//
2011-09-02 13:34:35 +02:00
// Each PML4 entry points to a page of Page Directory Pointer entires.
// So lets allocate space for them and fill them in in the IndexOfPdpEntries loop.
2011-07-12 22:35:46 +02:00
//
2011-09-02 13:34:35 +02:00
PageDirectoryPointerEntry = ( VOID * ) BigPageAddress ;
2011-09-13 07:48:57 +02:00
BigPageAddress + = SIZE_4KB ;
2011-07-12 22:35:46 +02:00
2011-09-02 13:34:35 +02:00
//
// Make a PML4 Entry
//
2017-02-26 18:43:04 +01:00
PageMapLevel4Entry - > Uint64 = ( UINT64 ) ( UINTN ) PageDirectoryPointerEntry | AddressEncMask ;
2011-09-02 13:34:35 +02:00
PageMapLevel4Entry - > Bits . ReadWrite = 1 ;
PageMapLevel4Entry - > Bits . Present = 1 ;
2011-09-13 07:48:57 +02:00
if ( Page1GSupport ) {
2011-09-19 15:02:05 +02:00
PageDirectory1GEntry = ( VOID * ) PageDirectoryPointerEntry ;
2018-06-27 15:08:52 +02:00
2011-09-13 07:48:57 +02:00
for ( IndexOfPageDirectoryEntries = 0 ; IndexOfPageDirectoryEntries < 512 ; IndexOfPageDirectoryEntries + + , PageDirectory1GEntry + + , PageAddress + = SIZE_1GB ) {
2011-07-12 22:35:46 +02:00
//
2011-09-02 13:34:35 +02:00
// Fill in the Page Directory entries
2011-07-12 22:35:46 +02:00
//
2017-02-26 18:43:04 +01:00
PageDirectory1GEntry - > Uint64 = ( UINT64 ) PageAddress | AddressEncMask ;
2011-09-13 07:48:57 +02:00
PageDirectory1GEntry - > Bits . ReadWrite = 1 ;
PageDirectory1GEntry - > Bits . Present = 1 ;
PageDirectory1GEntry - > Bits . MustBe1 = 1 ;
}
} else {
for ( IndexOfPdpEntries = 0 ; IndexOfPdpEntries < NumberOfPdpEntriesNeeded ; IndexOfPdpEntries + + , PageDirectoryPointerEntry + + ) {
//
// Each Directory Pointer entries points to a page of Page Directory entires.
// So allocate space for them and fill them in in the IndexOfPageDirectoryEntries loop.
2018-06-27 15:08:52 +02:00
//
2011-09-13 07:48:57 +02:00
PageDirectoryEntry = ( VOID * ) BigPageAddress ;
BigPageAddress + = SIZE_4KB ;
//
// Fill in a Page Directory Pointer Entries
//
2017-02-26 18:43:04 +01:00
PageDirectoryPointerEntry - > Uint64 = ( UINT64 ) ( UINTN ) PageDirectoryEntry | AddressEncMask ;
2011-09-13 07:48:57 +02:00
PageDirectoryPointerEntry - > Bits . ReadWrite = 1 ;
PageDirectoryPointerEntry - > Bits . Present = 1 ;
for ( IndexOfPageDirectoryEntries = 0 ; IndexOfPageDirectoryEntries < 512 ; IndexOfPageDirectoryEntries + + , PageDirectoryEntry + + , PageAddress + = SIZE_2MB ) {
//
// Fill in the Page Directory entries
//
2017-02-26 18:43:04 +01:00
PageDirectoryEntry - > Uint64 = ( UINT64 ) PageAddress | AddressEncMask ;
2011-09-13 07:48:57 +02:00
PageDirectoryEntry - > Bits . ReadWrite = 1 ;
PageDirectoryEntry - > Bits . Present = 1 ;
PageDirectoryEntry - > Bits . MustBe1 = 1 ;
}
}
2011-09-02 13:34:35 +02:00
2011-09-13 07:48:57 +02:00
for ( ; IndexOfPdpEntries < 512 ; IndexOfPdpEntries + + , PageDirectoryPointerEntry + + ) {
ZeroMem (
PageDirectoryPointerEntry ,
sizeof ( PAGE_MAP_AND_DIRECTORY_POINTER )
) ;
2011-07-12 22:35:46 +02:00
}
}
}
2011-09-02 13:34:35 +02:00
//
// For the PML4 entries we are not using fill in a null entry.
//
for ( ; IndexOfPml4Entries < 512 ; IndexOfPml4Entries + + , PageMapLevel4Entry + + ) {
2011-09-13 07:48:57 +02:00
ZeroMem (
PageMapLevel4Entry ,
sizeof ( PAGE_MAP_AND_DIRECTORY_POINTER )
) ;
2011-09-02 13:34:35 +02:00
}
}
/**
Return function from long mode to 32 - bit mode .
@ param EntrypointContext Context for mode switching
@ param ReturnContext Context for mode switching
* */
VOID
ReturnFunction (
SWITCH_32_TO_64_CONTEXT * EntrypointContext ,
SWITCH_64_TO_32_CONTEXT * ReturnContext
)
2015-07-27 05:04:41 +02:00
{
2011-09-02 13:34:35 +02:00
//
// Restore original GDT
//
AsmWriteGdtr ( & ReturnContext - > Gdtr ) ;
2015-07-27 05:04:41 +02:00
2011-09-02 13:34:35 +02:00
//
// return to original caller
//
LongJump ( ( BASE_LIBRARY_JUMP_BUFFER * ) ( UINTN ) EntrypointContext - > JumpBuffer , 1 ) ;
2015-07-27 05:04:41 +02:00
2011-09-02 13:34:35 +02:00
//
// never be here
2015-07-27 05:04:41 +02:00
//
2011-09-02 13:34:35 +02:00
ASSERT ( FALSE ) ;
}
/**
Thunk function from 32 - bit protection mode to long mode .
@ param PageTableAddress Page table base address
@ param Context Context for mode switching
@ param ReturnContext Context for mode switching
@ retval EFI_SUCCESS Function successfully executed .
* */
EFI_STATUS
Thunk32To64 (
EFI_PHYSICAL_ADDRESS PageTableAddress ,
SWITCH_32_TO_64_CONTEXT * Context ,
SWITCH_64_TO_32_CONTEXT * ReturnContext
)
{
UINTN SetJumpFlag ;
EFI_STATUS Status ;
//
// Save return address, LongJump will return here then
//
SetJumpFlag = SetJump ( ( BASE_LIBRARY_JUMP_BUFFER * ) ( UINTN ) Context - > JumpBuffer ) ;
if ( SetJumpFlag = = 0 ) {
2011-07-12 22:35:46 +02:00
//
2015-07-27 05:04:41 +02:00
// Build 4G Page Tables.
2011-07-12 22:35:46 +02:00
//
2015-07-27 05:04:41 +02:00
Create4GPageTables ( PageTableAddress , Context - > Page1GSupport ) ;
2011-09-02 13:34:35 +02:00
//
// Create 64-bit GDT
//
AsmWriteGdtr ( & mGdt ) ;
2011-07-12 22:35:46 +02:00
2011-09-02 13:34:35 +02:00
//
// Write CR3
//
AsmWriteCr3 ( ( UINTN ) PageTableAddress ) ;
2016-11-03 08:47:06 +01:00
DEBUG ( (
DEBUG_INFO ,
" %a() Stack Base: 0x%lx, Stack Size: 0x%lx \n " ,
2023-04-06 21:50:06 +02:00
__func__ ,
2016-11-03 08:47:06 +01:00
Context - > StackBufferBase ,
Context - > StackBufferLength
) ) ;
2012-08-27 10:36:23 +02:00
//
// Disable interrupt of Debug timer, since the IDT table cannot work in long mode
//
SaveAndSetDebugTimerInterrupt ( FALSE ) ;
2011-09-02 13:34:35 +02:00
//
// Transfer to long mode
//
AsmEnablePaging64 (
0x38 ,
( UINT64 ) Context - > EntryPoint ,
( UINT64 ) ( UINTN ) Context ,
( UINT64 ) ( UINTN ) ReturnContext ,
Context - > StackBufferBase + Context - > StackBufferLength
) ;
}
2015-07-27 05:04:41 +02:00
2011-09-02 13:34:35 +02:00
//
// Convert to 32-bit Status and return
//
Status = EFI_SUCCESS ;
if ( ( UINTN ) ReturnContext - > ReturnStatus ! = 0 ) {
Status = ENCODE_ERROR ( ( UINTN ) ReturnContext - > ReturnStatus ) ;
}
2018-06-27 15:08:52 +02:00
2011-09-02 13:34:35 +02:00
return Status ;
2011-07-12 22:35:46 +02:00
}
2011-09-02 13:34:35 +02:00
/**
If in 32 bit protection mode , and coalesce image is of X64 , switch to long mode .
@ param LongModeBuffer The context of long mode .
@ param CoalesceEntry Entry of coalesce image .
@ param BlockListAddr Address of block list .
2016-04-26 06:52:42 +02:00
@ param MemoryResource Pointer to the buffer of memory resource descriptor .
2011-09-02 13:34:35 +02:00
@ param MemoryBase Base of memory range .
@ param MemorySize Size of memory range .
@ retval EFI_SUCCESS Successfully switched to long mode and execute coalesce .
@ retval Others Failed to execute coalesce in long mode .
* */
EFI_STATUS
ModeSwitch (
IN EFI_CAPSULE_LONG_MODE_BUFFER * LongModeBuffer ,
IN COALESCE_ENTRY CoalesceEntry ,
IN EFI_PHYSICAL_ADDRESS BlockListAddr ,
2016-04-26 06:52:42 +02:00
IN MEMORY_RESOURCE_DESCRIPTOR * MemoryResource ,
2011-09-02 13:34:35 +02:00
IN OUT VOID * * MemoryBase ,
IN OUT UINTN * MemorySize
)
{
EFI_STATUS Status ;
EFI_PHYSICAL_ADDRESS MemoryBase64 ;
UINT64 MemorySize64 ;
EFI_PHYSICAL_ADDRESS MemoryEnd64 ;
SWITCH_32_TO_64_CONTEXT Context ;
SWITCH_64_TO_32_CONTEXT ReturnContext ;
BASE_LIBRARY_JUMP_BUFFER JumpBuffer ;
EFI_PHYSICAL_ADDRESS ReservedRangeBase ;
EFI_PHYSICAL_ADDRESS ReservedRangeEnd ;
2015-07-27 05:04:41 +02:00
BOOLEAN Page1GSupport ;
2011-09-02 13:34:35 +02:00
ZeroMem ( & Context , sizeof ( SWITCH_32_TO_64_CONTEXT ) ) ;
ZeroMem ( & ReturnContext , sizeof ( SWITCH_64_TO_32_CONTEXT ) ) ;
2018-06-27 15:08:52 +02:00
2011-09-02 13:34:35 +02:00
MemoryBase64 = ( UINT64 ) ( UINTN ) * MemoryBase ;
MemorySize64 = ( UINT64 ) ( UINTN ) * MemorySize ;
MemoryEnd64 = MemoryBase64 + MemorySize64 ;
2015-07-27 05:04:41 +02:00
Page1GSupport = IsPage1GSupport ( ) ;
2011-09-02 13:34:35 +02:00
//
2018-06-27 15:08:52 +02:00
// Merge memory range reserved for stack and page table
2011-09-02 13:34:35 +02:00
//
if ( LongModeBuffer - > StackBaseAddress < LongModeBuffer - > PageTableAddress ) {
ReservedRangeBase = LongModeBuffer - > StackBaseAddress ;
2015-07-27 05:04:41 +02:00
ReservedRangeEnd = LongModeBuffer - > PageTableAddress + CalculatePageTableSize ( Page1GSupport ) ;
2011-09-02 13:34:35 +02:00
} else {
ReservedRangeBase = LongModeBuffer - > PageTableAddress ;
ReservedRangeEnd = LongModeBuffer - > StackBaseAddress + LongModeBuffer - > StackSize ;
}
2015-07-27 05:04:41 +02:00
2011-09-02 13:34:35 +02:00
//
// Check if memory range reserved is overlap with MemoryBase ~ MemoryBase + MemorySize.
// If they are overlapped, get a larger range to process capsule data.
//
if ( ReservedRangeBase < = MemoryBase64 ) {
if ( ReservedRangeEnd < MemoryEnd64 ) {
MemoryBase64 = ReservedRangeEnd ;
} else {
2019-04-11 10:20:36 +02:00
DEBUG ( ( DEBUG_ERROR , " Memory is not enough to process capsule! \n " ) ) ;
2011-09-02 13:34:35 +02:00
return EFI_OUT_OF_RESOURCES ;
}
} else if ( ReservedRangeBase < MemoryEnd64 ) {
if ( ( ReservedRangeEnd < MemoryEnd64 ) & &
( ReservedRangeBase - MemoryBase64 < MemoryEnd64 - ReservedRangeEnd ) )
{
MemoryBase64 = ReservedRangeEnd ;
} else {
MemorySize64 = ( UINT64 ) ( UINTN ) ( ReservedRangeBase - MemoryBase64 ) ;
}
2015-07-27 05:04:41 +02:00
}
2011-09-02 13:34:35 +02:00
//
// Initialize context jumping to 64-bit enviroment
//
Context . JumpBuffer = ( EFI_PHYSICAL_ADDRESS ) ( UINTN ) & JumpBuffer ;
Context . StackBufferBase = LongModeBuffer - > StackBaseAddress ;
Context . StackBufferLength = LongModeBuffer - > StackSize ;
Context . EntryPoint = ( EFI_PHYSICAL_ADDRESS ) ( UINTN ) CoalesceEntry ;
Context . BlockListAddr = BlockListAddr ;
2016-04-26 06:52:42 +02:00
Context . MemoryResource = ( EFI_PHYSICAL_ADDRESS ) ( UINTN ) MemoryResource ;
2011-09-02 13:34:35 +02:00
Context . MemoryBase64Ptr = ( EFI_PHYSICAL_ADDRESS ) ( UINTN ) & MemoryBase64 ;
Context . MemorySize64Ptr = ( EFI_PHYSICAL_ADDRESS ) ( UINTN ) & MemorySize64 ;
2015-07-27 05:04:41 +02:00
Context . Page1GSupport = Page1GSupport ;
2017-02-26 18:43:04 +01:00
Context . AddressEncMask = PcdGet64 ( PcdPteMemoryEncryptionAddressOrMask ) & PAGING_1G_ADDRESS_MASK_64 ;
2011-09-02 13:34:35 +02:00
//
// Prepare data for return back
//
ReturnContext . ReturnCs = 0x10 ;
ReturnContext . ReturnEntryPoint = ( EFI_PHYSICAL_ADDRESS ) ( UINTN ) ReturnFunction ;
//
// Will save the return status of processing capsule
//
ReturnContext . ReturnStatus = 0 ;
2018-06-27 15:08:52 +02:00
2011-09-02 13:34:35 +02:00
//
// Save original GDT
//
AsmReadGdtr ( ( IA32_DESCRIPTOR * ) & ReturnContext . Gdtr ) ;
2018-06-27 15:08:52 +02:00
2011-09-02 13:34:35 +02:00
Status = Thunk32To64 ( LongModeBuffer - > PageTableAddress , & Context , & ReturnContext ) ;
2018-06-27 15:08:52 +02:00
2011-09-02 13:34:35 +02:00
if ( ! EFI_ERROR ( Status ) ) {
* MemoryBase = ( VOID * ) ( UINTN ) MemoryBase64 ;
* MemorySize = ( UINTN ) MemorySize64 ;
}
return Status ;
}
2011-07-12 22:35:46 +02:00
2011-09-05 11:53:37 +02:00
/**
Locates the coalesce image entry point , and detects its machine type .
@ param CoalesceImageEntryPoint Pointer to coalesce image entry point for output .
@ param CoalesceImageMachineType Pointer to machine type of coalesce image .
@ retval EFI_SUCCESS Coalesce image successfully located .
@ retval Others Failed to locate the coalesce image .
* */
EFI_STATUS
FindCapsuleCoalesceImage (
OUT EFI_PHYSICAL_ADDRESS * CoalesceImageEntryPoint ,
OUT UINT16 * CoalesceImageMachineType
)
{
EFI_STATUS Status ;
UINTN Instance ;
EFI_PEI_LOAD_FILE_PPI * LoadFile ;
EFI_PEI_FV_HANDLE VolumeHandle ;
EFI_PEI_FILE_HANDLE FileHandle ;
EFI_PHYSICAL_ADDRESS CoalesceImageAddress ;
UINT64 CoalesceImageSize ;
UINT32 AuthenticationState ;
Instance = 0 ;
while ( TRUE ) {
Status = PeiServicesFfsFindNextVolume ( Instance + + , & VolumeHandle ) ;
if ( EFI_ERROR ( Status ) ) {
return Status ;
}
2021-12-05 23:54:02 +01:00
2011-09-05 11:53:37 +02:00
Status = PeiServicesFfsFindFileByName ( PcdGetPtr ( PcdCapsuleCoalesceFile ) , VolumeHandle , & FileHandle ) ;
if ( ! EFI_ERROR ( Status ) ) {
Status = PeiServicesLocatePpi ( & gEfiPeiLoadFilePpiGuid , 0 , NULL , ( VOID * * ) & LoadFile ) ;
ASSERT_EFI_ERROR ( Status ) ;
Status = LoadFile - > LoadFile (
LoadFile ,
FileHandle ,
& CoalesceImageAddress ,
& CoalesceImageSize ,
CoalesceImageEntryPoint ,
& AuthenticationState
) ;
if ( EFI_ERROR ( Status ) ) {
2019-04-11 10:20:36 +02:00
DEBUG ( ( DEBUG_ERROR , " Unable to find PE32 section in CapsuleX64 image ffs %r! \n " , Status ) ) ;
2011-09-05 11:53:37 +02:00
return Status ;
}
2021-12-05 23:54:02 +01:00
2024-07-18 10:30:45 +02:00
* CoalesceImageMachineType = EFI_IMAGE_MACHINE_X64 ;
// FIXME: Get machine type from context once PPI is adapted
//*CoalesceImageMachineType = UefiImageLoaderGetMachineType ((VOID *)(UINTN)CoalesceImageAddress);
2011-09-05 11:53:37 +02:00
break ;
} else {
continue ;
}
}
return Status ;
}
2015-07-27 05:04:41 +02:00
/**
Gets the reserved long mode buffer .
@ param LongModeBuffer Pointer to the long mode buffer for output .
@ retval EFI_SUCCESS Long mode buffer successfully retrieved .
@ retval Others Variable storing long mode buffer not found .
* */
EFI_STATUS
GetLongModeContext (
OUT EFI_CAPSULE_LONG_MODE_BUFFER * LongModeBuffer
)
{
EFI_STATUS Status ;
UINTN Size ;
EFI_PEI_READ_ONLY_VARIABLE2_PPI * PPIVariableServices ;
Status = PeiServicesLocatePpi (
& gEfiPeiReadOnlyVariable2PpiGuid ,
0 ,
NULL ,
( VOID * * ) & PPIVariableServices
) ;
ASSERT_EFI_ERROR ( Status ) ;
Size = sizeof ( EFI_CAPSULE_LONG_MODE_BUFFER ) ;
Status = PPIVariableServices - > GetVariable (
PPIVariableServices ,
EFI_CAPSULE_LONG_MODE_BUFFER_NAME ,
& gEfiCapsuleVendorGuid ,
NULL ,
& Size ,
LongModeBuffer
) ;
if ( EFI_ERROR ( Status ) ) {
2019-04-11 10:20:36 +02:00
DEBUG ( ( DEBUG_ERROR , " Error Get LongModeBuffer variable %r! \n " , Status ) ) ;
2015-07-27 05:04:41 +02:00
}
2021-12-05 23:54:02 +01:00
2015-07-27 05:04:41 +02:00
return Status ;
}
2021-12-05 23:54:02 +01:00
2011-09-05 11:53:37 +02:00
# endif
2016-04-26 06:52:42 +02:00
# if defined (MDE_CPU_IA32) || defined (MDE_CPU_X64)
2021-12-05 23:54:02 +01:00
2016-04-26 06:52:42 +02:00
/**
Get physical address bits .
@ return Physical address bits .
* */
UINT8
GetPhysicalAddressBits (
VOID
)
{
UINT32 RegEax ;
UINT8 PhysicalAddressBits ;
VOID * Hob ;
//
// Get physical address bits supported.
//
Hob = GetFirstHob ( EFI_HOB_TYPE_CPU ) ;
if ( Hob ! = NULL ) {
PhysicalAddressBits = ( ( EFI_HOB_CPU * ) Hob ) - > SizeOfMemorySpace ;
} else {
AsmCpuid ( 0x80000000 , & RegEax , NULL , NULL , NULL ) ;
if ( RegEax > = 0x80000008 ) {
AsmCpuid ( 0x80000008 , & RegEax , NULL , NULL , NULL ) ;
PhysicalAddressBits = ( UINT8 ) RegEax ;
} else {
PhysicalAddressBits = 36 ;
}
}
//
// IA-32e paging translates 48-bit linear addresses to 52-bit physical addresses.
//
ASSERT ( PhysicalAddressBits < = 52 ) ;
if ( PhysicalAddressBits > 48 ) {
PhysicalAddressBits = 48 ;
}
return PhysicalAddressBits ;
}
2021-12-05 23:54:02 +01:00
2016-04-26 06:52:42 +02:00
# endif
2017-11-17 06:56:37 +01:00
/**
Sort memory resource entries based upon PhysicalStart , from low to high .
@ param [ in , out ] MemoryResource A pointer to the memory resource entry buffer .
* */
VOID
SortMemoryResourceDescriptor (
IN OUT MEMORY_RESOURCE_DESCRIPTOR * MemoryResource
)
{
MEMORY_RESOURCE_DESCRIPTOR * MemoryResourceEntry ;
MEMORY_RESOURCE_DESCRIPTOR * NextMemoryResourceEntry ;
MEMORY_RESOURCE_DESCRIPTOR TempMemoryResource ;
MemoryResourceEntry = MemoryResource ;
NextMemoryResourceEntry = MemoryResource + 1 ;
while ( MemoryResourceEntry - > ResourceLength ! = 0 ) {
while ( NextMemoryResourceEntry - > ResourceLength ! = 0 ) {
if ( MemoryResourceEntry - > PhysicalStart > NextMemoryResourceEntry - > PhysicalStart ) {
CopyMem ( & TempMemoryResource , MemoryResourceEntry , sizeof ( MEMORY_RESOURCE_DESCRIPTOR ) ) ;
CopyMem ( MemoryResourceEntry , NextMemoryResourceEntry , sizeof ( MEMORY_RESOURCE_DESCRIPTOR ) ) ;
CopyMem ( NextMemoryResourceEntry , & TempMemoryResource , sizeof ( MEMORY_RESOURCE_DESCRIPTOR ) ) ;
}
NextMemoryResourceEntry = NextMemoryResourceEntry + 1 ;
}
MemoryResourceEntry = MemoryResourceEntry + 1 ;
NextMemoryResourceEntry = MemoryResourceEntry + 1 ;
}
}
/**
Merge continous memory resource entries .
@ param [ in , out ] MemoryResource A pointer to the memory resource entry buffer .
* */
VOID
MergeMemoryResourceDescriptor (
IN OUT MEMORY_RESOURCE_DESCRIPTOR * MemoryResource
)
{
MEMORY_RESOURCE_DESCRIPTOR * MemoryResourceEntry ;
MEMORY_RESOURCE_DESCRIPTOR * NewMemoryResourceEntry ;
MEMORY_RESOURCE_DESCRIPTOR * NextMemoryResourceEntry ;
MEMORY_RESOURCE_DESCRIPTOR * MemoryResourceEnd ;
MemoryResourceEntry = MemoryResource ;
NewMemoryResourceEntry = MemoryResource ;
while ( MemoryResourceEntry - > ResourceLength ! = 0 ) {
CopyMem ( NewMemoryResourceEntry , MemoryResourceEntry , sizeof ( MEMORY_RESOURCE_DESCRIPTOR ) ) ;
NextMemoryResourceEntry = MemoryResourceEntry + 1 ;
while ( ( NextMemoryResourceEntry - > ResourceLength ! = 0 ) & &
( NextMemoryResourceEntry - > PhysicalStart = = ( MemoryResourceEntry - > PhysicalStart + MemoryResourceEntry - > ResourceLength ) ) )
{
MemoryResourceEntry - > ResourceLength + = NextMemoryResourceEntry - > ResourceLength ;
if ( NewMemoryResourceEntry ! = MemoryResourceEntry ) {
NewMemoryResourceEntry - > ResourceLength + = NextMemoryResourceEntry - > ResourceLength ;
}
2018-06-27 15:08:52 +02:00
2017-11-17 06:56:37 +01:00
NextMemoryResourceEntry = NextMemoryResourceEntry + 1 ;
}
MemoryResourceEntry = NextMemoryResourceEntry ;
NewMemoryResourceEntry = NewMemoryResourceEntry + 1 ;
}
//
// Set NULL terminate memory resource descriptor after merging.
//
MemoryResourceEnd = NewMemoryResourceEntry ;
ZeroMem ( MemoryResourceEnd , sizeof ( MEMORY_RESOURCE_DESCRIPTOR ) ) ;
}
2016-04-26 06:52:42 +02:00
/**
Build memory resource descriptor from resource descriptor in HOB list .
@ return Pointer to the buffer of memory resource descriptor .
NULL if no memory resource descriptor reported in HOB list
before capsule Coalesce .
* */
MEMORY_RESOURCE_DESCRIPTOR *
BuildMemoryResourceDescriptor (
VOID
)
{
EFI_PEI_HOB_POINTERS Hob ;
UINTN Index ;
EFI_HOB_RESOURCE_DESCRIPTOR * ResourceDescriptor ;
MEMORY_RESOURCE_DESCRIPTOR * MemoryResource ;
EFI_STATUS Status ;
//
// Get the count of memory resource descriptor.
//
Index = 0 ;
Hob . Raw = GetFirstHob ( EFI_HOB_TYPE_RESOURCE_DESCRIPTOR ) ;
while ( Hob . Raw ! = NULL ) {
ResourceDescriptor = ( EFI_HOB_RESOURCE_DESCRIPTOR * ) Hob . Raw ;
if ( ResourceDescriptor - > ResourceType = = EFI_RESOURCE_SYSTEM_MEMORY ) {
Index + + ;
}
2021-12-05 23:54:02 +01:00
2016-04-26 06:52:42 +02:00
Hob . Raw = GET_NEXT_HOB ( Hob ) ;
Hob . Raw = GetNextHob ( EFI_HOB_TYPE_RESOURCE_DESCRIPTOR , Hob . Raw ) ;
}
if ( Index = = 0 ) {
2019-04-11 10:20:36 +02:00
DEBUG ( ( DEBUG_INFO | DEBUG_WARN , " No memory resource descriptor reported in HOB list before capsule Coalesce \n " ) ) ;
2016-04-26 06:52:42 +02:00
# if defined (MDE_CPU_IA32) || defined (MDE_CPU_X64)
//
// Allocate memory to hold memory resource descriptor,
// include extra one NULL terminate memory resource descriptor.
//
Status = PeiServicesAllocatePool ( ( 1 + 1 ) * sizeof ( MEMORY_RESOURCE_DESCRIPTOR ) , ( VOID * * ) & MemoryResource ) ;
ASSERT_EFI_ERROR ( Status ) ;
ZeroMem ( MemoryResource , ( 1 + 1 ) * sizeof ( MEMORY_RESOURCE_DESCRIPTOR ) ) ;
2018-06-27 15:08:52 +02:00
2016-04-26 06:52:42 +02:00
MemoryResource [ 0 ] . PhysicalStart = 0 ;
MemoryResource [ 0 ] . ResourceLength = LShiftU64 ( 1 , GetPhysicalAddressBits ( ) ) ;
2019-04-11 10:20:36 +02:00
DEBUG ( (
DEBUG_INFO ,
" MemoryResource[0x0] - Start(0x%0lx) Length(0x%0lx) \n " ,
2016-04-26 06:52:42 +02:00
MemoryResource [ 0x0 ] . PhysicalStart ,
MemoryResource [ 0x0 ] . ResourceLength
) ) ;
return MemoryResource ;
# else
return NULL ;
# endif
}
//
// Allocate memory to hold memory resource descriptor,
// include extra one NULL terminate memory resource descriptor.
//
Status = PeiServicesAllocatePool ( ( Index + 1 ) * sizeof ( MEMORY_RESOURCE_DESCRIPTOR ) , ( VOID * * ) & MemoryResource ) ;
ASSERT_EFI_ERROR ( Status ) ;
ZeroMem ( MemoryResource , ( Index + 1 ) * sizeof ( MEMORY_RESOURCE_DESCRIPTOR ) ) ;
//
// Get the content of memory resource descriptor.
//
Index = 0 ;
Hob . Raw = GetFirstHob ( EFI_HOB_TYPE_RESOURCE_DESCRIPTOR ) ;
while ( Hob . Raw ! = NULL ) {
ResourceDescriptor = ( EFI_HOB_RESOURCE_DESCRIPTOR * ) Hob . Raw ;
if ( ResourceDescriptor - > ResourceType = = EFI_RESOURCE_SYSTEM_MEMORY ) {
2019-04-11 10:20:36 +02:00
DEBUG ( (
DEBUG_INFO ,
" MemoryResource[0x%x] - Start(0x%0lx) Length(0x%0lx) \n " ,
2016-04-26 06:52:42 +02:00
Index ,
ResourceDescriptor - > PhysicalStart ,
ResourceDescriptor - > ResourceLength
) ) ;
MemoryResource [ Index ] . PhysicalStart = ResourceDescriptor - > PhysicalStart ;
MemoryResource [ Index ] . ResourceLength = ResourceDescriptor - > ResourceLength ;
Index + + ;
}
2021-12-05 23:54:02 +01:00
2016-04-26 06:52:42 +02:00
Hob . Raw = GET_NEXT_HOB ( Hob ) ;
Hob . Raw = GetNextHob ( EFI_HOB_TYPE_RESOURCE_DESCRIPTOR , Hob . Raw ) ;
}
2017-11-17 06:56:37 +01:00
SortMemoryResourceDescriptor ( MemoryResource ) ;
MergeMemoryResourceDescriptor ( MemoryResource ) ;
DEBUG ( ( DEBUG_INFO , " Dump MemoryResource[] after sorted and merged \n " ) ) ;
for ( Index = 0 ; MemoryResource [ Index ] . ResourceLength ! = 0 ; Index + + ) {
DEBUG ( (
DEBUG_INFO ,
" MemoryResource[0x%x] - Start(0x%0lx) Length(0x%0lx) \n " ,
Index ,
MemoryResource [ Index ] . PhysicalStart ,
MemoryResource [ Index ] . ResourceLength
) ) ;
}
2016-04-26 06:52:42 +02:00
return MemoryResource ;
}
2011-07-12 22:35:46 +02:00
/**
2019-06-04 04:24:02 +02:00
Check if the capsules are staged .
2011-07-12 22:35:46 +02:00
2019-06-10 03:43:59 +02:00
@ retval TRUE The capsules are staged .
@ retval FALSE The capsules are not staged .
2011-07-12 22:35:46 +02:00
* */
2019-06-10 03:43:59 +02:00
BOOLEAN
2019-06-10 03:38:18 +02:00
AreCapsulesStaged (
2019-06-10 03:43:59 +02:00
VOID
2019-06-04 04:24:02 +02:00
)
{
EFI_STATUS Status ;
UINTN Size ;
EFI_PEI_READ_ONLY_VARIABLE2_PPI * PPIVariableServices ;
2019-06-10 03:43:59 +02:00
EFI_PHYSICAL_ADDRESS CapsuleDataPtr64 ;
2019-06-04 04:24:02 +02:00
2019-06-10 03:43:59 +02:00
CapsuleDataPtr64 = 0 ;
2019-06-04 04:24:02 +02:00
Status = PeiServicesLocatePpi (
& gEfiPeiReadOnlyVariable2PpiGuid ,
0 ,
NULL ,
( VOID * * ) & PPIVariableServices
) ;
if ( EFI_ERROR ( Status ) ) {
DEBUG ( ( DEBUG_ERROR , " Failed to find ReadOnlyVariable2PPI \n " ) ) ;
2019-06-10 03:43:59 +02:00
return FALSE ;
2019-06-04 04:24:02 +02:00
}
//
// Check for Update capsule
//
Size = sizeof ( CapsuleDataPtr64 ) ;
Status = PPIVariableServices - > GetVariable (
PPIVariableServices ,
EFI_CAPSULE_VARIABLE_NAME ,
& gEfiCapsuleVendorGuid ,
NULL ,
& Size ,
( VOID * ) & CapsuleDataPtr64
) ;
if ( ! EFI_ERROR ( Status ) ) {
2019-06-10 03:43:59 +02:00
return TRUE ;
2019-06-04 04:24:02 +02:00
}
2019-06-10 03:43:59 +02:00
return FALSE ;
2019-06-04 04:24:02 +02:00
}
/**
Check all the variables for SG list heads and get the count and addresses .
@ param ListLength A pointer would return the SG list length .
@ param HeadList A ponter to the capsule SG list .
@ retval EFI_SUCCESS a valid capsule is present
@ retval EFI_NOT_FOUND if a valid capsule is not present
@ retval EFI_INVALID_PARAMETER the input parameter is invalid
@ retval EFI_OUT_OF_RESOURCES fail to allocate memory
* */
EFI_STATUS
2019-06-10 03:38:18 +02:00
GetScatterGatherHeadEntries (
2019-06-04 04:24:02 +02:00
OUT UINTN * ListLength ,
OUT EFI_PHYSICAL_ADDRESS * * HeadList
2011-07-12 22:35:46 +02:00
)
{
2019-06-10 03:52:22 +02:00
EFI_STATUS Status ;
UINTN Size ;
UINTN Index ;
UINTN TempIndex ;
UINTN ValidIndex ;
BOOLEAN Flag ;
CHAR16 CapsuleVarName [ 30 ] ;
CHAR16 * TempVarName ;
EFI_PHYSICAL_ADDRESS CapsuleDataPtr64 ;
EFI_PEI_READ_ONLY_VARIABLE2_PPI * PPIVariableServices ;
EFI_PHYSICAL_ADDRESS * TempList ;
EFI_PHYSICAL_ADDRESS * EnlargedTempList ;
UINTN TempListLength ;
2011-07-12 22:35:46 +02:00
Index = 0 ;
TempVarName = NULL ;
CapsuleVarName [ 0 ] = 0 ;
ValidIndex = 0 ;
2014-07-24 07:58:50 +02:00
CapsuleDataPtr64 = 0 ;
2018-06-27 15:08:52 +02:00
2019-06-04 04:24:02 +02:00
if ( ( ListLength = = NULL ) | | ( HeadList = = NULL ) ) {
2023-04-06 21:50:06 +02:00
DEBUG ( ( DEBUG_ERROR , " %a Invalid parameters. Inputs can't be NULL \n " , __func__ ) ) ;
2019-06-04 04:24:02 +02:00
ASSERT ( ListLength ! = NULL ) ;
ASSERT ( HeadList ! = NULL ) ;
return EFI_INVALID_PARAMETER ;
}
* ListLength = 0 ;
* HeadList = NULL ;
2011-07-12 22:35:46 +02:00
Status = PeiServicesLocatePpi (
& gEfiPeiReadOnlyVariable2PpiGuid ,
0 ,
NULL ,
2019-06-04 04:24:02 +02:00
( VOID * * ) & PPIVariableServices
2011-07-12 22:35:46 +02:00
) ;
2019-06-04 04:24:02 +02:00
if ( EFI_ERROR ( Status ) ) {
DEBUG ( ( DEBUG_ERROR , " Failed to find ReadOnlyVariable2PPI \n " ) ) ;
return Status ;
}
2019-06-10 03:52:22 +02:00
//
// Allocate memory for sg list head
//
TempListLength = DEFAULT_SG_LIST_HEADS * sizeof ( EFI_PHYSICAL_ADDRESS ) ;
TempList = AllocateZeroPool ( TempListLength ) ;
if ( TempList = = NULL ) {
DEBUG ( ( DEBUG_ERROR , " Failed to allocate memory \n " ) ) ;
return EFI_OUT_OF_RESOURCES ;
}
2019-06-04 04:24:02 +02:00
//
// setup var name buffer for update capsules
//
StrCpyS ( CapsuleVarName , sizeof ( CapsuleVarName ) / sizeof ( CHAR16 ) , EFI_CAPSULE_VARIABLE_NAME ) ;
TempVarName = CapsuleVarName + StrLen ( CapsuleVarName ) ;
2019-06-10 03:52:22 +02:00
while ( TRUE ) {
2019-06-04 04:24:02 +02:00
if ( Index ! = 0 ) {
UnicodeValueToStringS (
TempVarName ,
2019-06-10 03:54:52 +02:00
( sizeof ( CapsuleVarName ) - ( ( UINTN ) TempVarName - ( UINTN ) CapsuleVarName ) ) ,
2019-06-04 04:24:02 +02:00
0 ,
Index ,
0
) ;
}
2021-12-05 23:54:02 +01:00
2011-07-12 22:35:46 +02:00
Size = sizeof ( CapsuleDataPtr64 ) ;
2019-06-04 04:24:02 +02:00
Status = PPIVariableServices - > GetVariable (
PPIVariableServices ,
CapsuleVarName ,
& gEfiCapsuleVendorGuid ,
NULL ,
& Size ,
( VOID * ) & CapsuleDataPtr64
) ;
2018-06-27 15:08:52 +02:00
2019-06-04 04:24:02 +02:00
if ( EFI_ERROR ( Status ) ) {
if ( Status ! = EFI_NOT_FOUND ) {
2022-08-02 19:28:21 +02:00
DEBUG ( ( DEBUG_ERROR , " Unexpected error getting Capsule Update variable. Status = %r \n " , Status ) ) ;
2011-07-12 22:35:46 +02:00
}
2021-12-05 23:54:02 +01:00
2019-06-04 04:24:02 +02:00
break ;
}
2018-06-27 15:08:52 +02:00
2019-06-04 04:24:02 +02:00
//
// If this BlockList has been linked before, skip this variable
//
Flag = FALSE ;
for ( TempIndex = 0 ; TempIndex < ValidIndex ; TempIndex + + ) {
if ( TempList [ TempIndex ] = = CapsuleDataPtr64 ) {
Flag = TRUE ;
break ;
}
2011-07-12 22:35:46 +02:00
}
2021-12-05 23:54:02 +01:00
2019-06-04 04:24:02 +02:00
if ( Flag ) {
Index + + ;
continue ;
}
2019-06-10 03:52:22 +02:00
//
// The TempList is full, enlarge it
//
if ( ( ValidIndex + 1 ) > = TempListLength ) {
EnlargedTempList = AllocateZeroPool ( TempListLength * 2 ) ;
2019-06-25 05:22:49 +02:00
if ( EnlargedTempList = = NULL ) {
DEBUG ( ( DEBUG_ERROR , " Fail to allocate memory! \n " ) ) ;
return EFI_OUT_OF_RESOURCES ;
}
2021-12-05 23:54:02 +01:00
2019-06-10 03:52:22 +02:00
CopyMem ( EnlargedTempList , TempList , TempListLength ) ;
FreePool ( TempList ) ;
TempList = EnlargedTempList ;
TempListLength * = 2 ;
}
2019-06-04 04:24:02 +02:00
//
// add it to the cached list
//
TempList [ ValidIndex + + ] = CapsuleDataPtr64 ;
Index + + ;
}
if ( ValidIndex = = 0 ) {
2023-04-06 21:50:06 +02:00
DEBUG ( ( DEBUG_ERROR , " %a didn't find any SG lists in variables \n " , __func__ ) ) ;
2019-06-04 04:24:02 +02:00
return EFI_NOT_FOUND ;
}
* HeadList = AllocateZeroPool ( ( ValidIndex + 1 ) * sizeof ( EFI_PHYSICAL_ADDRESS ) ) ;
if ( * HeadList = = NULL ) {
DEBUG ( ( DEBUG_ERROR , " Failed to allocate memory \n " ) ) ;
return EFI_OUT_OF_RESOURCES ;
2011-07-12 22:35:46 +02:00
}
2018-06-27 15:08:52 +02:00
2019-06-04 04:24:02 +02:00
CopyMem ( * HeadList , TempList , ( ValidIndex ) * sizeof ( EFI_PHYSICAL_ADDRESS ) ) ;
* ListLength = ValidIndex ;
2011-07-12 22:35:46 +02:00
return EFI_SUCCESS ;
}
/**
Capsule PPI service to coalesce a fragmented capsule in memory .
@ param PeiServices General purpose services available to every PEIM .
@ param MemoryBase Pointer to the base of a block of memory that we can walk
all over while trying to coalesce our buffers .
On output , this variable will hold the base address of
a coalesced capsule .
@ param MemorySize Size of the memory region pointed to by MemoryBase .
On output , this variable will contain the size of the
coalesced capsule .
@ retval EFI_NOT_FOUND if we can ' t determine the boot mode
if the boot mode is not flash - update
if we could not find the capsule descriptors
@ retval EFI_BUFFER_TOO_SMALL
if we could not coalesce the capsule in the memory
region provided to us
@ retval EFI_SUCCESS if there ' s no capsule , or if we processed the
capsule successfully .
* */
EFI_STATUS
EFIAPI
CapsuleCoalesce (
IN EFI_PEI_SERVICES * * PeiServices ,
IN OUT VOID * * MemoryBase ,
IN OUT UINTN * MemorySize
)
{
2011-09-02 13:34:35 +02:00
EFI_STATUS Status ;
EFI_BOOT_MODE BootMode ;
2019-06-04 04:24:02 +02:00
UINTN ListLength ;
2011-09-02 13:34:35 +02:00
EFI_PHYSICAL_ADDRESS * VariableArrayAddress ;
2016-04-26 06:52:42 +02:00
MEMORY_RESOURCE_DESCRIPTOR * MemoryResource ;
2021-12-05 23:54:02 +01:00
2011-09-02 13:34:35 +02:00
# ifdef MDE_CPU_IA32
UINT16 CoalesceImageMachineType ;
EFI_PHYSICAL_ADDRESS CoalesceImageEntryPoint ;
COALESCE_ENTRY CoalesceEntry ;
EFI_CAPSULE_LONG_MODE_BUFFER LongModeBuffer ;
# endif
2021-12-05 23:54:02 +01:00
2019-06-04 04:24:02 +02:00
ListLength = 0 ;
VariableArrayAddress = NULL ;
2011-07-12 22:35:46 +02:00
//
// Someone should have already ascertained the boot mode. If it's not
// capsule update, then return normally.
//
Status = PeiServicesGetBootMode ( & BootMode ) ;
if ( EFI_ERROR ( Status ) | | ( BootMode ! = BOOT_ON_FLASH_UPDATE ) ) {
2019-04-11 10:20:36 +02:00
DEBUG ( ( DEBUG_ERROR , " Boot mode is not correct for capsule update path. \n " ) ) ;
2011-09-02 13:34:35 +02:00
Status = EFI_NOT_FOUND ;
goto Done ;
2011-07-12 22:35:46 +02:00
}
2018-06-27 15:08:52 +02:00
2011-07-12 22:35:46 +02:00
//
2019-06-04 04:24:02 +02:00
// Get SG list entries
2011-07-12 22:35:46 +02:00
//
2019-06-04 04:24:02 +02:00
Status = GetScatterGatherHeadEntries ( & ListLength , & VariableArrayAddress ) ;
2019-06-25 05:22:49 +02:00
if ( EFI_ERROR ( Status ) | | ( VariableArrayAddress = = NULL ) ) {
2023-04-06 21:50:06 +02:00
DEBUG ( ( DEBUG_ERROR , " %a failed to get Scatter Gather List Head Entries. Status = %r \n " , __func__ , Status ) ) ;
2011-09-02 13:34:35 +02:00
goto Done ;
2011-07-12 22:35:46 +02:00
}
2016-04-26 06:52:42 +02:00
MemoryResource = BuildMemoryResourceDescriptor ( ) ;
2011-09-02 13:34:35 +02:00
# ifdef MDE_CPU_IA32
if ( FeaturePcdGet ( PcdDxeIplSwitchToLongMode ) ) {
2011-07-12 22:35:46 +02:00
//
2011-09-02 13:34:35 +02:00
// Switch to 64-bit mode to process capsule data when:
// 1. When DXE phase is 64-bit
// 2. When the buffer for 64-bit transition exists
// 3. When Capsule X64 image is built in BIOS image
// In 64-bit mode, we can process capsule data above 4GB.
2011-07-12 22:35:46 +02:00
//
2011-09-02 13:34:35 +02:00
CoalesceImageEntryPoint = 0 ;
Status = GetLongModeContext ( & LongModeBuffer ) ;
if ( EFI_ERROR ( Status ) ) {
2019-04-11 10:20:36 +02:00
DEBUG ( ( DEBUG_ERROR , " Fail to find the variable for long mode context! \n " ) ) ;
2011-09-02 13:34:35 +02:00
Status = EFI_NOT_FOUND ;
goto Done ;
2011-07-12 22:35:46 +02:00
}
2018-06-27 15:08:52 +02:00
2011-09-02 13:34:35 +02:00
Status = FindCapsuleCoalesceImage ( & CoalesceImageEntryPoint , & CoalesceImageMachineType ) ;
if ( ( EFI_ERROR ( Status ) ) | | ( CoalesceImageMachineType ! = EFI_IMAGE_MACHINE_X64 ) ) {
2019-04-11 10:20:36 +02:00
DEBUG ( ( DEBUG_ERROR , " Fail to find CapsuleX64 module in FV! \n " ) ) ;
2011-09-02 13:34:35 +02:00
Status = EFI_NOT_FOUND ;
goto Done ;
2011-07-12 22:35:46 +02:00
}
2021-12-05 23:54:02 +01:00
2011-09-02 13:34:35 +02:00
ASSERT ( CoalesceImageEntryPoint ! = 0 ) ;
CoalesceEntry = ( COALESCE_ENTRY ) ( UINTN ) CoalesceImageEntryPoint ;
2016-04-26 06:52:42 +02:00
Status = ModeSwitch ( & LongModeBuffer , CoalesceEntry , ( EFI_PHYSICAL_ADDRESS ) ( UINTN ) VariableArrayAddress , MemoryResource , MemoryBase , MemorySize ) ;
2011-09-02 13:34:35 +02:00
} else {
2011-07-12 22:35:46 +02:00
//
2011-09-02 13:34:35 +02:00
// Capsule is processed in IA32 mode.
2011-07-12 22:35:46 +02:00
//
2016-04-26 06:52:42 +02:00
Status = CapsuleDataCoalesce ( PeiServices , ( EFI_PHYSICAL_ADDRESS * ) ( UINTN ) VariableArrayAddress , MemoryResource , MemoryBase , MemorySize ) ;
2011-07-12 22:35:46 +02:00
}
2021-12-05 23:54:02 +01:00
2011-09-02 13:34:35 +02:00
# else
2011-07-12 22:35:46 +02:00
//
2011-09-02 13:34:35 +02:00
// Process capsule directly.
2011-07-12 22:35:46 +02:00
//
2016-04-26 06:52:42 +02:00
Status = CapsuleDataCoalesce ( PeiServices , ( EFI_PHYSICAL_ADDRESS * ) ( UINTN ) VariableArrayAddress , MemoryResource , MemoryBase , MemorySize ) ;
2011-09-02 13:34:35 +02:00
# endif
2018-06-27 15:08:52 +02:00
2019-04-11 10:20:36 +02:00
DEBUG ( ( DEBUG_INFO , " Capsule Coalesce Status = %r! \n " , Status ) ) ;
2011-07-12 22:35:46 +02:00
2011-09-02 13:34:35 +02:00
if ( Status = = EFI_BUFFER_TOO_SMALL ) {
2019-04-11 10:20:36 +02:00
DEBUG ( ( DEBUG_ERROR , " There is not enough memory to process capsule! \n " ) ) ;
2011-09-02 13:34:35 +02:00
}
2018-06-27 15:08:52 +02:00
2011-09-02 13:34:35 +02:00
if ( Status = = EFI_NOT_FOUND ) {
2019-04-11 10:20:36 +02:00
DEBUG ( ( DEBUG_ERROR , " Fail to parse capsule descriptor in memory! \n " ) ) ;
2011-09-02 13:34:35 +02:00
REPORT_STATUS_CODE (
EFI_ERROR_CODE | EFI_ERROR_MAJOR ,
( EFI_SOFTWARE_PEI_MODULE | EFI_SW_PEI_EC_INVALID_CAPSULE_DESCRIPTOR )
) ;
}
2011-07-12 22:35:46 +02:00
2011-09-02 13:34:35 +02:00
Done :
2011-07-12 22:35:46 +02:00
return Status ;
}
/**
Determine if we ' re in capsule update boot mode .
@ param PeiServices PEI services table
@ retval EFI_SUCCESS if we have a capsule available
@ retval EFI_NOT_FOUND no capsule detected
* */
EFI_STATUS
EFIAPI
CheckCapsuleUpdate (
IN EFI_PEI_SERVICES * * PeiServices
)
{
2019-06-10 03:43:59 +02:00
if ( AreCapsulesStaged ( ) ) {
return EFI_SUCCESS ;
} else {
return EFI_NOT_FOUND ;
2019-06-04 04:24:02 +02:00
}
2011-07-12 22:35:46 +02:00
}
2021-12-05 23:54:02 +01:00
2011-07-12 22:35:46 +02:00
/**
2018-06-27 15:08:52 +02:00
This function will look at a capsule and determine if it ' s a test pattern .
2011-07-12 22:35:46 +02:00
If it is , then it will verify it and emit an error message if corruption is detected .
2018-06-27 15:08:52 +02:00
2011-07-12 22:35:46 +02:00
@ param PeiServices Standard pei services pointer
@ param CapsuleBase Base address of coalesced capsule , which is preceeded
by private data . Very implementation specific .
@ retval TRUE Capsule image is the test image
@ retval FALSE Capsule image is not the test image .
* */
BOOLEAN
CapsuleTestPattern (
IN EFI_PEI_SERVICES * * PeiServices ,
IN VOID * CapsuleBase
)
{
UINT32 * TestPtr ;
UINT32 TestCounter ;
UINT32 TestSize ;
BOOLEAN RetValue ;
RetValue = FALSE ;
//
// Look at the capsule data and determine if it's a test pattern. If it
// is, then test it now.
//
TestPtr = ( UINT32 * ) CapsuleBase ;
2011-09-02 13:34:35 +02:00
//
// 0x54534554 "TEST"
//
if ( * TestPtr = = 0x54534554 ) {
2011-07-12 22:35:46 +02:00
RetValue = TRUE ;
2019-04-11 10:20:36 +02:00
DEBUG ( ( DEBUG_INFO , " Capsule test pattern mode activated... \n " ) ) ;
2011-07-12 22:35:46 +02:00
TestSize = TestPtr [ 1 ] / sizeof ( UINT32 ) ;
//
// Skip over the signature and the size fields in the pattern data header
//
TestPtr + = 2 ;
TestCounter = 0 ;
while ( TestSize > 0 ) {
if ( * TestPtr ! = TestCounter ) {
2019-04-11 10:20:36 +02:00
DEBUG ( ( DEBUG_INFO , " Capsule test pattern mode FAILED: BaseAddr/FailAddr 0x%X 0x%X \n " , ( UINT32 ) ( UINTN ) ( EFI_CAPSULE_PEIM_PRIVATE_DATA * ) CapsuleBase , ( UINT32 ) ( UINTN ) TestPtr ) ) ;
2011-07-12 22:35:46 +02:00
return TRUE ;
}
TestPtr + + ;
TestCounter + + ;
TestSize - - ;
}
2019-04-11 10:20:36 +02:00
DEBUG ( ( DEBUG_INFO , " Capsule test pattern mode SUCCESS \n " ) ) ;
2011-07-12 22:35:46 +02:00
}
return RetValue ;
}
/**
Capsule PPI service that gets called after memory is available . The
capsule coalesce function , which must be called first , returns a base
address and size , which can be anything actually . Once the memory init
PEIM has discovered memory , then it should call this function and pass in
the base address and size returned by the coalesce function . Then this
function can create a capsule HOB and return .
@ param PeiServices standard pei services pointer
@ param CapsuleBase address returned by the capsule coalesce function . Most
likely this will actually be a pointer to private data .
@ param CapsuleSize value returned by the capsule coalesce function .
@ retval EFI_VOLUME_CORRUPTED CapsuleBase does not appear to point to a
coalesced capsule
@ retval EFI_SUCCESS if all goes well .
* */
EFI_STATUS
EFIAPI
CreateState (
IN EFI_PEI_SERVICES * * PeiServices ,
IN VOID * CapsuleBase ,
IN UINTN CapsuleSize
)
{
EFI_STATUS Status ;
EFI_CAPSULE_PEIM_PRIVATE_DATA * PrivateData ;
UINTN Size ;
EFI_PHYSICAL_ADDRESS NewBuffer ;
2014-01-17 04:51:01 +01:00
UINTN CapsuleNumber ;
2011-07-12 22:35:46 +02:00
UINT32 Index ;
EFI_PHYSICAL_ADDRESS BaseAddress ;
UINT64 Length ;
2021-12-05 23:54:02 +01:00
2011-07-12 22:35:46 +02:00
PrivateData = ( EFI_CAPSULE_PEIM_PRIVATE_DATA * ) CapsuleBase ;
if ( PrivateData - > Signature ! = EFI_CAPSULE_PEIM_PRIVATE_DATA_SIGNATURE ) {
return EFI_VOLUME_CORRUPTED ;
}
2021-12-05 23:54:02 +01:00
2014-01-17 04:51:01 +01:00
if ( PrivateData - > CapsuleAllImageSize > = MAX_ADDRESS ) {
2019-04-11 10:20:36 +02:00
DEBUG ( ( DEBUG_ERROR , " CapsuleAllImageSize too big - 0x%lx \n " , PrivateData - > CapsuleAllImageSize ) ) ;
2014-01-17 04:51:01 +01:00
return EFI_OUT_OF_RESOURCES ;
}
2021-12-05 23:54:02 +01:00
2014-01-17 04:51:01 +01:00
if ( PrivateData - > CapsuleNumber > = MAX_ADDRESS ) {
2019-04-11 10:20:36 +02:00
DEBUG ( ( DEBUG_ERROR , " CapsuleNumber too big - 0x%lx \n " , PrivateData - > CapsuleNumber ) ) ;
2014-01-17 04:51:01 +01:00
return EFI_OUT_OF_RESOURCES ;
}
2021-12-05 23:54:02 +01:00
2011-07-12 22:35:46 +02:00
//
// Capsule Number and Capsule Offset is in the tail of Capsule data.
//
2014-01-17 04:51:01 +01:00
Size = ( UINTN ) PrivateData - > CapsuleAllImageSize ;
CapsuleNumber = ( UINTN ) PrivateData - > CapsuleNumber ;
2011-07-12 22:35:46 +02:00
//
// Allocate the memory so that it gets preserved into DXE
//
Status = PeiServicesAllocatePages (
EfiRuntimeServicesData ,
EFI_SIZE_TO_PAGES ( Size ) ,
& NewBuffer
) ;
if ( Status ! = EFI_SUCCESS ) {
2019-04-11 10:20:36 +02:00
DEBUG ( ( DEBUG_ERROR , " AllocatePages Failed! \n " ) ) ;
2011-07-12 22:35:46 +02:00
return Status ;
}
2021-12-05 23:54:02 +01:00
2011-07-12 22:35:46 +02:00
//
// Copy to our new buffer for DXE
//
2019-04-11 10:20:36 +02:00
DEBUG ( ( DEBUG_INFO , " Capsule copy from 0x%8X to 0x%8X with size 0x%8X \n " , ( UINTN ) ( ( UINT8 * ) PrivateData + sizeof ( EFI_CAPSULE_PEIM_PRIVATE_DATA ) + ( CapsuleNumber - 1 ) * sizeof ( UINT64 ) ) , ( UINTN ) NewBuffer , Size ) ) ;
2014-01-17 04:51:01 +01:00
CopyMem ( ( VOID * ) ( UINTN ) NewBuffer , ( VOID * ) ( UINTN ) ( ( UINT8 * ) PrivateData + sizeof ( EFI_CAPSULE_PEIM_PRIVATE_DATA ) + ( CapsuleNumber - 1 ) * sizeof ( UINT64 ) ) , Size ) ;
2011-07-12 22:35:46 +02:00
//
// Check for test data pattern. If it is the test pattern, then we'll
2016-04-26 03:18:39 +02:00
// test it and still create the HOB so that it can be used to verify
2011-07-12 22:35:46 +02:00
// that capsules don't get corrupted all the way into BDS. BDS will
// still try to turn it into a firmware volume, but will think it's
// corrupted so nothing will happen.
//
DEBUG_CODE (
CapsuleTestPattern ( PeiServices , ( VOID * ) ( UINTN ) NewBuffer ) ;
) ;
//
// Build the UEFI Capsule Hob for each capsule image.
//
for ( Index = 0 ; Index < CapsuleNumber ; Index + + ) {
2014-01-17 04:51:01 +01:00
BaseAddress = NewBuffer + PrivateData - > CapsuleOffset [ Index ] ;
2011-07-12 22:35:46 +02:00
Length = ( ( EFI_CAPSULE_HEADER * ) ( ( UINTN ) BaseAddress ) ) - > CapsuleImageSize ;
BuildCvHob ( BaseAddress , Length ) ;
}
2018-06-27 15:08:52 +02:00
2011-07-12 22:35:46 +02:00
return EFI_SUCCESS ;
}
2015-05-08 09:43:35 +02:00
CONST EFI_PEI_CAPSULE_PPI mCapsulePpi = {
2011-07-12 22:35:46 +02:00
CapsuleCoalesce ,
CheckCapsuleUpdate ,
CreateState
} ;
CONST EFI_PEI_PPI_DESCRIPTOR mUefiPpiListCapsule = {
( EFI_PEI_PPI_DESCRIPTOR_PPI | EFI_PEI_PPI_DESCRIPTOR_TERMINATE_LIST ) ,
2015-05-08 09:43:35 +02:00
& gEfiPeiCapsulePpiGuid ,
( EFI_PEI_CAPSULE_PPI * ) & mCapsulePpi
2011-07-12 22:35:46 +02:00
} ;
/**
Entry point function for the PEIM
@ param FileHandle Handle of the file being invoked .
@ param PeiServices Describes the list of possible PEI Services .
@ return EFI_SUCCESS If we installed our PPI
* */
EFI_STATUS
EFIAPI
CapsuleMain (
IN EFI_PEI_FILE_HANDLE FileHandle ,
IN CONST EFI_PEI_SERVICES * * PeiServices
)
{
//
// Just produce our PPI
//
return PeiServicesInstallPpi ( & mUefiPpiListCapsule ) ;
}