UefiCpuPkg: Use Top of each AP's stack to save CpuMpData

To remove the dependency of CPU register, 4/8 byte at the top of the
stack is occupied for CpuMpData. BIST information is also taken care
here. This modification is only for PEI phase, since in DXE phase
CpuMpData is accessed via global variable.

Signed-off-by: Yuanhao Xie <yuanhao.xie@intel.com>
Cc: Eric Dong <eric.dong@intel.com>
Reviewed-by: Ray Ni <ray.ni@intel.com>
Cc: Rahul Kumar <rahul1.kumar@intel.com>
This commit is contained in:
Yuanhao Xie 2022-08-19 14:17:28 +08:00 committed by mergify[bot]
parent 76cf3d35e6
commit 9ab2b34dd4
5 changed files with 59 additions and 13 deletions

View File

@ -179,6 +179,14 @@ ProgramStack:
mov esp, dword [edi + CPU_INFO_IN_HOB.ApTopOfStack] mov esp, dword [edi + CPU_INFO_IN_HOB.ApTopOfStack]
CProcedureInvoke: CProcedureInvoke:
;
; Reserve 4 bytes for CpuMpData.
; When the AP wakes up again via INIT-SIPI-SIPI, push 0 will cause the existing CpuMpData to be overwritten with 0.
; CpuMpData is filled in via InitializeApData() during the first time INIT-SIPI-SIPI,
; while overwirrten may occurs when under ApInHltLoop but InitFlag is not set to ApInitConfig.
; Therefore reservation is implemented by sub esp instead of push 0.
;
sub esp, 4
push ebp ; push BIST data at top of AP stack push ebp ; push BIST data at top of AP stack
xor ebp, ebp ; clear ebp for call stack trace xor ebp, ebp ; clear ebp for call stack trace
push ebp push ebp

View File

@ -599,6 +599,7 @@ InitializeApData (
{ {
CPU_INFO_IN_HOB *CpuInfoInHob; CPU_INFO_IN_HOB *CpuInfoInHob;
MSR_IA32_PLATFORM_ID_REGISTER PlatformIdMsr; MSR_IA32_PLATFORM_ID_REGISTER PlatformIdMsr;
AP_STACK_DATA *ApStackData;
CpuInfoInHob = (CPU_INFO_IN_HOB *)(UINTN)CpuMpData->CpuInfoInHob; CpuInfoInHob = (CPU_INFO_IN_HOB *)(UINTN)CpuMpData->CpuInfoInHob;
CpuInfoInHob[ProcessorNumber].InitialApicId = GetInitialApicId (); CpuInfoInHob[ProcessorNumber].InitialApicId = GetInitialApicId ();
@ -606,6 +607,12 @@ InitializeApData (
CpuInfoInHob[ProcessorNumber].Health = BistData; CpuInfoInHob[ProcessorNumber].Health = BistData;
CpuInfoInHob[ProcessorNumber].ApTopOfStack = ApTopOfStack; CpuInfoInHob[ProcessorNumber].ApTopOfStack = ApTopOfStack;
//
// AP_STACK_DATA is stored at the top of AP Stack
//
ApStackData = (AP_STACK_DATA *)((UINTN)ApTopOfStack - sizeof (AP_STACK_DATA));
ApStackData->MpData = CpuMpData;
CpuMpData->CpuData[ProcessorNumber].Waiting = FALSE; CpuMpData->CpuData[ProcessorNumber].Waiting = FALSE;
CpuMpData->CpuData[ProcessorNumber].CpuHealthy = (BistData == 0) ? TRUE : FALSE; CpuMpData->CpuData[ProcessorNumber].CpuHealthy = (BistData == 0) ? TRUE : FALSE;
@ -651,6 +658,7 @@ ApWakeupFunction (
CPU_INFO_IN_HOB *CpuInfoInHob; CPU_INFO_IN_HOB *CpuInfoInHob;
UINT64 ApTopOfStack; UINT64 ApTopOfStack;
UINTN CurrentApicMode; UINTN CurrentApicMode;
AP_STACK_DATA *ApStackData;
// //
// AP finished assembly code and begin to execute C code // AP finished assembly code and begin to execute C code
@ -676,7 +684,9 @@ ApWakeupFunction (
// This is first time AP wakeup, get BIST information from AP stack // This is first time AP wakeup, get BIST information from AP stack
// //
ApTopOfStack = CpuMpData->Buffer + (ProcessorNumber + 1) * CpuMpData->CpuApStackSize; ApTopOfStack = CpuMpData->Buffer + (ProcessorNumber + 1) * CpuMpData->CpuApStackSize;
BistData = *(UINT32 *)((UINTN)ApTopOfStack - sizeof (UINTN)); ApStackData = (AP_STACK_DATA *)((UINTN)ApTopOfStack - sizeof (AP_STACK_DATA));
BistData = (UINT32)ApStackData->Bist;
// //
// CpuMpData->CpuData[0].VolatileRegisters is initialized based on BSP environment, // CpuMpData->CpuData[0].VolatileRegisters is initialized based on BSP environment,
// to initialize AP in InitConfig path. // to initialize AP in InitConfig path.
@ -1824,6 +1834,10 @@ MpInitLibInitialize (
AsmGetAddressMap (&AddressMap); AsmGetAddressMap (&AddressMap);
GetApResetVectorSize (&AddressMap, &ApResetVectorSizeBelow1Mb, &ApResetVectorSizeAbove1Mb); GetApResetVectorSize (&AddressMap, &ApResetVectorSizeBelow1Mb, &ApResetVectorSizeAbove1Mb);
ApStackSize = PcdGet32 (PcdCpuApStackSize); ApStackSize = PcdGet32 (PcdCpuApStackSize);
//
// ApStackSize must be power of 2
//
ASSERT ((ApStackSize & (ApStackSize - 1)) == 0);
ApLoopMode = GetApLoopMode (&MonitorFilterSize); ApLoopMode = GetApLoopMode (&MonitorFilterSize);
// //
@ -1832,6 +1846,10 @@ MpInitLibInitialize (
SaveVolatileRegisters (&VolatileRegisters); SaveVolatileRegisters (&VolatileRegisters);
BufferSize = ApStackSize * MaxLogicalProcessorNumber; BufferSize = ApStackSize * MaxLogicalProcessorNumber;
//
// Allocate extra ApStackSize to let AP stack align on ApStackSize bounday
//
BufferSize += ApStackSize;
BufferSize += MonitorFilterSize * MaxLogicalProcessorNumber; BufferSize += MonitorFilterSize * MaxLogicalProcessorNumber;
BufferSize += ApResetVectorSizeBelow1Mb; BufferSize += ApResetVectorSizeBelow1Mb;
BufferSize = ALIGN_VALUE (BufferSize, 8); BufferSize = ALIGN_VALUE (BufferSize, 8);
@ -1841,13 +1859,13 @@ MpInitLibInitialize (
MpBuffer = AllocatePages (EFI_SIZE_TO_PAGES (BufferSize)); MpBuffer = AllocatePages (EFI_SIZE_TO_PAGES (BufferSize));
ASSERT (MpBuffer != NULL); ASSERT (MpBuffer != NULL);
ZeroMem (MpBuffer, BufferSize); ZeroMem (MpBuffer, BufferSize);
Buffer = (UINTN)MpBuffer; Buffer = ALIGN_VALUE ((UINTN)MpBuffer, ApStackSize);
// //
// The layout of the Buffer is as below: // The layout of the Buffer is as below (lower address on top):
// //
// +--------------------+ <-- Buffer // +--------------------+ <-- Buffer (Pointer of CpuMpData is stored in the top of each AP's stack.)
// AP Stacks (N) // AP Stacks (N) (StackTop = (RSP + ApStackSize) & ~ApStackSize))
// +--------------------+ <-- MonitorBuffer // +--------------------+ <-- MonitorBuffer
// AP Monitor Filters (N) // AP Monitor Filters (N)
// +--------------------+ <-- BackupBufferAddr (CpuMpData->BackupBuffer) // +--------------------+ <-- BackupBufferAddr (CpuMpData->BackupBuffer)
@ -1855,7 +1873,7 @@ MpInitLibInitialize (
// +--------------------+ // +--------------------+
// Padding // Padding
// +--------------------+ <-- ApIdtBase (8-byte boundary) // +--------------------+ <-- ApIdtBase (8-byte boundary)
// AP IDT All APs share one separate IDT. So AP can get address of CPU_MP_DATA from IDT Base. // AP IDT All APs share one separate IDT.
// +--------------------+ <-- CpuMpData // +--------------------+ <-- CpuMpData
// CPU_MP_DATA // CPU_MP_DATA
// +--------------------+ <-- CpuMpData->CpuData // +--------------------+ <-- CpuMpData->CpuData
@ -1892,10 +1910,11 @@ MpInitLibInitialize (
// //
// Make sure no memory usage outside of the allocated buffer. // Make sure no memory usage outside of the allocated buffer.
// (ApStackSize - (Buffer - (UINTN)MpBuffer)) is the redundant caused by alignment
// //
ASSERT ( ASSERT (
(CpuMpData->CpuInfoInHob + sizeof (CPU_INFO_IN_HOB) * MaxLogicalProcessorNumber) == (CpuMpData->CpuInfoInHob + sizeof (CPU_INFO_IN_HOB) * MaxLogicalProcessorNumber) ==
Buffer + BufferSize (UINTN)MpBuffer + BufferSize - (ApStackSize - Buffer + (UINTN)MpBuffer)
); );
// //

View File

@ -301,6 +301,14 @@ struct _CPU_MP_DATA {
UINT64 GhcbBase; UINT64 GhcbBase;
}; };
//
// AP_STACK_DATA is stored at the top of each AP stack.
//
typedef struct {
UINTN Bist;
CPU_MP_DATA *MpData;
} AP_STACK_DATA;
#define AP_SAFE_STACK_SIZE 128 #define AP_SAFE_STACK_SIZE 128
#define AP_RESET_STACK_SIZE AP_SAFE_STACK_SIZE #define AP_RESET_STACK_SIZE AP_SAFE_STACK_SIZE

View File

@ -89,7 +89,7 @@ EnableDebugAgent (
/** /**
Get pointer to CPU MP Data structure. Get pointer to CPU MP Data structure.
For BSP, the pointer is retrieved from HOB. For BSP, the pointer is retrieved from HOB.
For AP, the structure is just after IDT. For AP, the structure is stored in the top of each AP's stack.
@return The pointer to CPU MP Data structure. @return The pointer to CPU MP Data structure.
**/ **/
@ -100,15 +100,17 @@ GetCpuMpData (
{ {
CPU_MP_DATA *CpuMpData; CPU_MP_DATA *CpuMpData;
MSR_IA32_APIC_BASE_REGISTER ApicBaseMsr; MSR_IA32_APIC_BASE_REGISTER ApicBaseMsr;
IA32_DESCRIPTOR Idtr; UINTN ApTopOfStack;
AP_STACK_DATA *ApStackData;
ApicBaseMsr.Uint64 = AsmReadMsr64 (MSR_IA32_APIC_BASE); ApicBaseMsr.Uint64 = AsmReadMsr64 (MSR_IA32_APIC_BASE);
if (ApicBaseMsr.Bits.BSP == 1) { if (ApicBaseMsr.Bits.BSP == 1) {
CpuMpData = GetCpuMpDataFromGuidedHob (); CpuMpData = GetCpuMpDataFromGuidedHob ();
ASSERT (CpuMpData != NULL); ASSERT (CpuMpData != NULL);
} else { } else {
AsmReadIdtr (&Idtr); ApTopOfStack = ALIGN_VALUE ((UINTN)&ApTopOfStack, (UINTN)PcdGet32 (PcdCpuApStackSize));
CpuMpData = (CPU_MP_DATA *)(Idtr.Base + Idtr.Limit + 1); ApStackData = (AP_STACK_DATA *)((UINTN)ApTopOfStack- sizeof (AP_STACK_DATA));
CpuMpData = (CPU_MP_DATA *)ApStackData->MpData;
} }
return CpuMpData; return CpuMpData;

View File

@ -239,11 +239,20 @@ ProgramStack:
mov rsp, qword [rdi + CPU_INFO_IN_HOB.ApTopOfStack] mov rsp, qword [rdi + CPU_INFO_IN_HOB.ApTopOfStack]
CProcedureInvoke: CProcedureInvoke:
;
; Reserve 8 bytes for CpuMpData.
; When the AP wakes up again via INIT-SIPI-SIPI, push 0 will cause the existing CpuMpData to be overwritten with 0.
; CpuMpData is filled in via InitializeApData() during the first time INIT-SIPI-SIPI,
; while overwirrten may occurs when under ApInHltLoop but InitFlag is not set to ApInitConfig.
; Therefore reservation is implemented by sub rsp instead of push 0.
;
sub rsp, 8
push rbp ; Push BIST data at top of AP stack push rbp ; Push BIST data at top of AP stack
xor rbp, rbp ; Clear ebp for call stack trace xor rbp, rbp ; Clear ebp for call stack trace
push rbp push rbp
mov rbp, rsp mov rbp, rsp
push qword 0 ; Push 8 bytes for alignment
mov rax, qword [esi + MP_CPU_EXCHANGE_INFO_FIELD (InitializeFloatingPointUnits)] mov rax, qword [esi + MP_CPU_EXCHANGE_INFO_FIELD (InitializeFloatingPointUnits)]
sub rsp, 20h sub rsp, 20h
call rax ; Call assembly function to initialize FPU per UEFI spec call rax ; Call assembly function to initialize FPU per UEFI spec