]> git.proxmox.com Git - mirror_edk2.git/blame - UefiCpuPkg/PiSmmCpuDxeSmm/CpuS3.c
UefiCpuPkg/PiSmmCpuDxeSmm: Remove debug message.
[mirror_edk2.git] / UefiCpuPkg / PiSmmCpuDxeSmm / CpuS3.c
CommitLineData
529a5a86
MK
1/** @file\r
2Code for Processor S3 restoration\r
3\r
02031cfc 4Copyright (c) 2006 - 2019, Intel Corporation. All rights reserved.<BR>\r
0acd8697 5SPDX-License-Identifier: BSD-2-Clause-Patent\r
529a5a86
MK
6\r
7**/\r
8\r
9#include "PiSmmCpuDxeSmm.h"\r
10\r
e21e355e 11#pragma pack(1)\r
529a5a86
MK
12typedef struct {\r
13 UINTN Lock;\r
14 VOID *StackStart;\r
15 UINTN StackSize;\r
16 VOID *ApFunction;\r
17 IA32_DESCRIPTOR GdtrProfile;\r
18 IA32_DESCRIPTOR IdtrProfile;\r
19 UINT32 BufferStart;\r
20 UINT32 Cr3;\r
e21e355e 21 UINTN InitializeFloatingPointUnitsAddress;\r
529a5a86 22} MP_CPU_EXCHANGE_INFO;\r
e21e355e 23#pragma pack()\r
529a5a86
MK
24\r
25typedef struct {\r
26 UINT8 *RendezvousFunnelAddress;\r
27 UINTN PModeEntryOffset;\r
28 UINTN FlatJumpOffset;\r
29 UINTN Size;\r
30 UINTN LModeEntryOffset;\r
31 UINTN LongJumpOffset;\r
32} MP_ASSEMBLY_ADDRESS_MAP;\r
33\r
6c4c15fa 34//\r
93324390 35// Flags used when program the register.\r
6c4c15fa 36//\r
93324390 37typedef struct {\r
9bae7811
ED
38 volatile UINTN ConsoleLogLock; // Spinlock used to control console.\r
39 volatile UINTN MemoryMappedLock; // Spinlock used to program mmio\r
40 volatile UINT32 *CoreSemaphoreCount; // Semaphore container used to program\r
41 // core level semaphore.\r
42 volatile UINT32 *PackageSemaphoreCount; // Semaphore container used to program\r
43 // package level semaphore.\r
93324390 44} PROGRAM_CPU_REGISTER_FLAGS;\r
6c4c15fa 45\r
7677b4db
ED
46//\r
47// Signal that SMM BASE relocation is complete.\r
48//\r
49volatile BOOLEAN mInitApsAfterSmmBaseReloc;\r
50\r
529a5a86
MK
51/**\r
52 Get starting address and size of the rendezvous entry for APs.\r
53 Information for fixing a jump instruction in the code is also returned.\r
54\r
55 @param AddressMap Output buffer for address map information.\r
56**/\r
57VOID *\r
58EFIAPI\r
59AsmGetAddressMap (\r
60 MP_ASSEMBLY_ADDRESS_MAP *AddressMap\r
61 );\r
62\r
63#define LEGACY_REGION_SIZE (2 * 0x1000)\r
64#define LEGACY_REGION_BASE (0xA0000 - LEGACY_REGION_SIZE)\r
529a5a86 65\r
93324390 66PROGRAM_CPU_REGISTER_FLAGS mCpuFlags;\r
529a5a86 67ACPI_CPU_DATA mAcpiCpuData;\r
c773514d 68volatile UINT32 mNumberToFinish;\r
529a5a86
MK
69MP_CPU_EXCHANGE_INFO *mExchangeInfo;\r
70BOOLEAN mRestoreSmmConfigurationInS3 = FALSE;\r
529a5a86 71\r
0bdc9e75
SZ
72//\r
73// S3 boot flag\r
74//\r
75BOOLEAN mSmmS3Flag = FALSE;\r
76\r
77//\r
78// Pointer to structure used during S3 Resume\r
79//\r
80SMM_S3_RESUME_STATE *mSmmS3ResumeState = NULL;\r
81\r
b10d5ddc
SZ
82BOOLEAN mAcpiS3Enable = TRUE;\r
83\r
4a0f88dd
JF
84UINT8 *mApHltLoopCode = NULL;\r
85UINT8 mApHltLoopCodeTemplate[] = {\r
ec8a3877
JF
86 0x8B, 0x44, 0x24, 0x04, // mov eax, dword ptr [esp+4]\r
87 0xF0, 0xFF, 0x08, // lock dec dword ptr [eax]\r
88 0xFA, // cli\r
89 0xF4, // hlt\r
90 0xEB, 0xFC // jmp $-2\r
4a0f88dd
JF
91 };\r
92\r
529a5a86
MK
93/**\r
94 Sync up the MTRR values for all processors.\r
95\r
96 @param MtrrTable Table holding fixed/variable MTRR values to be loaded.\r
97**/\r
98VOID\r
99EFIAPI\r
100LoadMtrrData (\r
101 EFI_PHYSICAL_ADDRESS MtrrTable\r
102 )\r
103/*++\r
104\r
105Routine Description:\r
106\r
107 Sync up the MTRR values for all processors.\r
108\r
109Arguments:\r
110\r
111Returns:\r
112 None\r
113\r
114--*/\r
115{\r
116 MTRR_SETTINGS *MtrrSettings;\r
117\r
118 MtrrSettings = (MTRR_SETTINGS *) (UINTN) MtrrTable;\r
119 MtrrSetAllMtrrs (MtrrSettings);\r
120}\r
121\r
122/**\r
93324390 123 Increment semaphore by 1.\r
529a5a86 124\r
93324390 125 @param Sem IN: 32-bit unsigned integer\r
529a5a86 126\r
93324390
ED
127**/\r
128VOID\r
129S3ReleaseSemaphore (\r
130 IN OUT volatile UINT32 *Sem\r
131 )\r
132{\r
133 InterlockedIncrement (Sem);\r
134}\r
135\r
136/**\r
137 Decrement the semaphore by 1 if it is not zero.\r
529a5a86 138\r
93324390
ED
139 Performs an atomic decrement operation for semaphore.\r
140 The compare exchange operation must be performed using\r
141 MP safe mechanisms.\r
142\r
143 @param Sem IN: 32-bit unsigned integer\r
144\r
145**/\r
146VOID\r
147S3WaitForSemaphore (\r
148 IN OUT volatile UINT32 *Sem\r
149 )\r
150{\r
151 UINT32 Value;\r
152\r
153 do {\r
154 Value = *Sem;\r
155 } while (Value == 0 ||\r
156 InterlockedCompareExchange32 (\r
157 Sem,\r
158 Value,\r
159 Value - 1\r
160 ) != Value);\r
161}\r
162\r
163/**\r
164 Initialize the CPU registers from a register table.\r
165\r
166 @param[in] RegisterTable The register table for this AP.\r
167 @param[in] ApLocation AP location info for this ap.\r
168 @param[in] CpuStatus CPU status info for this CPU.\r
169 @param[in] CpuFlags Flags data structure used when program the register.\r
170\r
171 @note This service could be called by BSP/APs.\r
529a5a86
MK
172**/\r
173VOID\r
93324390
ED
174ProgramProcessorRegister (\r
175 IN CPU_REGISTER_TABLE *RegisterTable,\r
176 IN EFI_CPU_PHYSICAL_LOCATION *ApLocation,\r
177 IN CPU_STATUS_INFORMATION *CpuStatus,\r
178 IN PROGRAM_CPU_REGISTER_FLAGS *CpuFlags\r
529a5a86
MK
179 )\r
180{\r
181 CPU_REGISTER_TABLE_ENTRY *RegisterTableEntry;\r
182 UINTN Index;\r
183 UINTN Value;\r
93324390
ED
184 CPU_REGISTER_TABLE_ENTRY *RegisterTableEntryHead;\r
185 volatile UINT32 *SemaphorePtr;\r
186 UINT32 FirstThread;\r
187 UINT32 PackageThreadsCount;\r
188 UINT32 CurrentThread;\r
189 UINTN ProcessorIndex;\r
93324390
ED
190 UINTN ValidThreadCount;\r
191 UINT32 *ValidCoreCountPerPackage;\r
529a5a86
MK
192\r
193 //\r
194 // Traverse Register Table of this logical processor\r
195 //\r
93324390
ED
196 RegisterTableEntryHead = (CPU_REGISTER_TABLE_ENTRY *) (UINTN) RegisterTable->RegisterTableEntry;\r
197\r
198 for (Index = 0; Index < RegisterTable->TableLength; Index++) {\r
199\r
200 RegisterTableEntry = &RegisterTableEntryHead[Index];\r
201\r
529a5a86
MK
202 //\r
203 // Check the type of specified register\r
204 //\r
205 switch (RegisterTableEntry->RegisterType) {\r
206 //\r
207 // The specified register is Control Register\r
208 //\r
209 case ControlRegister:\r
210 switch (RegisterTableEntry->Index) {\r
211 case 0:\r
212 Value = AsmReadCr0 ();\r
213 Value = (UINTN) BitFieldWrite64 (\r
214 Value,\r
215 RegisterTableEntry->ValidBitStart,\r
216 RegisterTableEntry->ValidBitStart + RegisterTableEntry->ValidBitLength - 1,\r
217 (UINTN) RegisterTableEntry->Value\r
218 );\r
219 AsmWriteCr0 (Value);\r
220 break;\r
221 case 2:\r
222 Value = AsmReadCr2 ();\r
223 Value = (UINTN) BitFieldWrite64 (\r
224 Value,\r
225 RegisterTableEntry->ValidBitStart,\r
226 RegisterTableEntry->ValidBitStart + RegisterTableEntry->ValidBitLength - 1,\r
227 (UINTN) RegisterTableEntry->Value\r
228 );\r
229 AsmWriteCr2 (Value);\r
230 break;\r
231 case 3:\r
232 Value = AsmReadCr3 ();\r
233 Value = (UINTN) BitFieldWrite64 (\r
234 Value,\r
235 RegisterTableEntry->ValidBitStart,\r
236 RegisterTableEntry->ValidBitStart + RegisterTableEntry->ValidBitLength - 1,\r
237 (UINTN) RegisterTableEntry->Value\r
238 );\r
239 AsmWriteCr3 (Value);\r
240 break;\r
241 case 4:\r
242 Value = AsmReadCr4 ();\r
243 Value = (UINTN) BitFieldWrite64 (\r
244 Value,\r
245 RegisterTableEntry->ValidBitStart,\r
246 RegisterTableEntry->ValidBitStart + RegisterTableEntry->ValidBitLength - 1,\r
247 (UINTN) RegisterTableEntry->Value\r
248 );\r
249 AsmWriteCr4 (Value);\r
250 break;\r
251 default:\r
252 break;\r
253 }\r
254 break;\r
255 //\r
256 // The specified register is Model Specific Register\r
257 //\r
258 case Msr:\r
259 //\r
260 // If this function is called to restore register setting after INIT signal,\r
261 // there is no need to restore MSRs in register table.\r
262 //\r
263 if (RegisterTableEntry->ValidBitLength >= 64) {\r
264 //\r
265 // If length is not less than 64 bits, then directly write without reading\r
266 //\r
267 AsmWriteMsr64 (\r
268 RegisterTableEntry->Index,\r
269 RegisterTableEntry->Value\r
270 );\r
271 } else {\r
529a5a86
MK
272 //\r
273 // Set the bit section according to bit start and length\r
274 //\r
275 AsmMsrBitFieldWrite64 (\r
276 RegisterTableEntry->Index,\r
277 RegisterTableEntry->ValidBitStart,\r
278 RegisterTableEntry->ValidBitStart + RegisterTableEntry->ValidBitLength - 1,\r
279 RegisterTableEntry->Value\r
280 );\r
529a5a86
MK
281 }\r
282 break;\r
283 //\r
6c4c15fa
JF
284 // MemoryMapped operations\r
285 //\r
286 case MemoryMapped:\r
93324390 287 AcquireSpinLock (&CpuFlags->MemoryMappedLock);\r
6c4c15fa 288 MmioBitFieldWrite32 (\r
30b7a50b 289 (UINTN)(RegisterTableEntry->Index | LShiftU64 (RegisterTableEntry->HighIndex, 32)),\r
6c4c15fa
JF
290 RegisterTableEntry->ValidBitStart,\r
291 RegisterTableEntry->ValidBitStart + RegisterTableEntry->ValidBitLength - 1,\r
292 (UINT32)RegisterTableEntry->Value\r
293 );\r
93324390 294 ReleaseSpinLock (&CpuFlags->MemoryMappedLock);\r
6c4c15fa
JF
295 break;\r
296 //\r
529a5a86
MK
297 // Enable or disable cache\r
298 //\r
299 case CacheControl:\r
300 //\r
301 // If value of the entry is 0, then disable cache. Otherwise, enable cache.\r
302 //\r
303 if (RegisterTableEntry->Value == 0) {\r
304 AsmDisableCache ();\r
305 } else {\r
306 AsmEnableCache ();\r
307 }\r
308 break;\r
309\r
93324390
ED
310 case Semaphore:\r
311 // Semaphore works logic like below:\r
312 //\r
313 // V(x) = LibReleaseSemaphore (Semaphore[FirstThread + x]);\r
314 // P(x) = LibWaitForSemaphore (Semaphore[FirstThread + x]);\r
315 //\r
316 // All threads (T0...Tn) waits in P() line and continues running\r
317 // together.\r
318 //\r
319 //\r
320 // T0 T1 ... Tn\r
321 //\r
322 // V(0...n) V(0...n) ... V(0...n)\r
323 // n * P(0) n * P(1) ... n * P(n)\r
324 //\r
325 ASSERT (\r
e07e3ceb 326 (ApLocation != NULL) &&\r
93324390 327 (CpuStatus->ValidCoreCountPerPackage != 0) &&\r
9bae7811
ED
328 (CpuFlags->CoreSemaphoreCount != NULL) &&\r
329 (CpuFlags->PackageSemaphoreCount != NULL)\r
93324390 330 );\r
93324390
ED
331 switch (RegisterTableEntry->Value) {\r
332 case CoreDepType:\r
9bae7811 333 SemaphorePtr = CpuFlags->CoreSemaphoreCount;\r
93324390
ED
334 //\r
335 // Get Offset info for the first thread in the core which current thread belongs to.\r
336 //\r
337 FirstThread = (ApLocation->Package * CpuStatus->MaxCoreCount + ApLocation->Core) * CpuStatus->MaxThreadCount;\r
338 CurrentThread = FirstThread + ApLocation->Thread;\r
339 //\r
340 // First Notify all threads in current Core that this thread has ready.\r
341 //\r
342 for (ProcessorIndex = 0; ProcessorIndex < CpuStatus->MaxThreadCount; ProcessorIndex ++) {\r
343 S3ReleaseSemaphore (&SemaphorePtr[FirstThread + ProcessorIndex]);\r
344 }\r
345 //\r
346 // Second, check whether all valid threads in current core have ready.\r
347 //\r
348 for (ProcessorIndex = 0; ProcessorIndex < CpuStatus->MaxThreadCount; ProcessorIndex ++) {\r
349 S3WaitForSemaphore (&SemaphorePtr[CurrentThread]);\r
350 }\r
351 break;\r
352\r
353 case PackageDepType:\r
9bae7811 354 SemaphorePtr = CpuFlags->PackageSemaphoreCount;\r
93324390
ED
355 ValidCoreCountPerPackage = (UINT32 *)(UINTN)CpuStatus->ValidCoreCountPerPackage;\r
356 //\r
357 // Get Offset info for the first thread in the package which current thread belongs to.\r
358 //\r
359 FirstThread = ApLocation->Package * CpuStatus->MaxCoreCount * CpuStatus->MaxThreadCount;\r
360 //\r
361 // Get the possible threads count for current package.\r
362 //\r
363 PackageThreadsCount = CpuStatus->MaxThreadCount * CpuStatus->MaxCoreCount;\r
364 CurrentThread = FirstThread + CpuStatus->MaxThreadCount * ApLocation->Core + ApLocation->Thread;\r
365 //\r
366 // Get the valid thread count for current package.\r
367 //\r
368 ValidThreadCount = CpuStatus->MaxThreadCount * ValidCoreCountPerPackage[ApLocation->Package];\r
369\r
370 //\r
371 // Different packages may have different valid cores in them. If driver maintail clearly\r
372 // cores number in different packages, the logic will be much complicated.\r
373 // Here driver just simply records the max core number in all packages and use it as expect\r
374 // core number for all packages.\r
375 // In below two steps logic, first current thread will Release semaphore for each thread\r
376 // in current package. Maybe some threads are not valid in this package, but driver don't\r
377 // care. Second, driver will let current thread wait semaphore for all valid threads in\r
378 // current package. Because only the valid threads will do release semaphore for this\r
379 // thread, driver here only need to wait the valid thread count.\r
380 //\r
381\r
382 //\r
383 // First Notify all threads in current package that this thread has ready.\r
384 //\r
385 for (ProcessorIndex = 0; ProcessorIndex < PackageThreadsCount ; ProcessorIndex ++) {\r
386 S3ReleaseSemaphore (&SemaphorePtr[FirstThread + ProcessorIndex]);\r
387 }\r
388 //\r
389 // Second, check whether all valid threads in current package have ready.\r
390 //\r
391 for (ProcessorIndex = 0; ProcessorIndex < ValidThreadCount; ProcessorIndex ++) {\r
392 S3WaitForSemaphore (&SemaphorePtr[CurrentThread]);\r
393 }\r
394 break;\r
395\r
396 default:\r
397 break;\r
398 }\r
399 break;\r
400\r
529a5a86
MK
401 default:\r
402 break;\r
403 }\r
404 }\r
405}\r
406\r
93324390
ED
407/**\r
408\r
409 Set Processor register for one AP.\r
e07e3ceb 410\r
93324390
ED
411 @param PreSmmRegisterTable Use pre Smm register table or register table.\r
412\r
413**/\r
414VOID\r
415SetRegister (\r
416 IN BOOLEAN PreSmmRegisterTable\r
417 )\r
418{\r
419 CPU_REGISTER_TABLE *RegisterTable;\r
420 CPU_REGISTER_TABLE *RegisterTables;\r
421 UINT32 InitApicId;\r
422 UINTN ProcIndex;\r
423 UINTN Index;\r
424\r
425 if (PreSmmRegisterTable) {\r
426 RegisterTables = (CPU_REGISTER_TABLE *)(UINTN)mAcpiCpuData.PreSmmInitRegisterTable;\r
427 } else {\r
428 RegisterTables = (CPU_REGISTER_TABLE *)(UINTN)mAcpiCpuData.RegisterTable;\r
429 }\r
430\r
431 InitApicId = GetInitialApicId ();\r
432 RegisterTable = NULL;\r
7db4034f 433 ProcIndex = (UINTN)-1;\r
93324390
ED
434 for (Index = 0; Index < mAcpiCpuData.NumberOfCpus; Index++) {\r
435 if (RegisterTables[Index].InitialApicId == InitApicId) {\r
436 RegisterTable = &RegisterTables[Index];\r
437 ProcIndex = Index;\r
438 break;\r
439 }\r
440 }\r
441 ASSERT (RegisterTable != NULL);\r
442\r
443 if (mAcpiCpuData.ApLocation != 0) {\r
444 ProgramProcessorRegister (\r
445 RegisterTable,\r
446 (EFI_CPU_PHYSICAL_LOCATION *)(UINTN)mAcpiCpuData.ApLocation + ProcIndex,\r
447 &mAcpiCpuData.CpuStatus,\r
448 &mCpuFlags\r
449 );\r
450 } else {\r
451 ProgramProcessorRegister (\r
452 RegisterTable,\r
453 NULL,\r
454 &mAcpiCpuData.CpuStatus,\r
455 &mCpuFlags\r
456 );\r
457 }\r
458}\r
459\r
529a5a86 460/**\r
7677b4db 461 AP initialization before then after SMBASE relocation in the S3 boot path.\r
529a5a86
MK
462**/\r
463VOID\r
94744aa2 464InitializeAp (\r
529a5a86
MK
465 VOID\r
466 )\r
467{\r
7677b4db
ED
468 UINTN TopOfStack;\r
469 UINT8 Stack[128];\r
529a5a86
MK
470\r
471 LoadMtrrData (mAcpiCpuData.MtrrTable);\r
472\r
93324390 473 SetRegister (TRUE);\r
7677b4db 474\r
529a5a86
MK
475 //\r
476 // Count down the number with lock mechanism.\r
477 //\r
478 InterlockedDecrement (&mNumberToFinish);\r
529a5a86 479\r
7677b4db
ED
480 //\r
481 // Wait for BSP to signal SMM Base relocation done.\r
482 //\r
483 while (!mInitApsAfterSmmBaseReloc) {\r
484 CpuPause ();\r
485 }\r
529a5a86
MK
486\r
487 ProgramVirtualWireMode ();\r
488 DisableLvtInterrupts ();\r
489\r
93324390 490 SetRegister (FALSE);\r
529a5a86
MK
491\r
492 //\r
ec8a3877 493 // Place AP into the safe code, count down the number with lock mechanism in the safe code.\r
4a0f88dd 494 //\r
672b80c8
MK
495 TopOfStack = (UINTN) Stack + sizeof (Stack);\r
496 TopOfStack &= ~(UINTN) (CPU_STACK_ALIGNMENT - 1);\r
4a0f88dd 497 CopyMem ((VOID *) (UINTN) mApHltLoopCode, mApHltLoopCodeTemplate, sizeof (mApHltLoopCodeTemplate));\r
672b80c8 498 TransferApToSafeState ((UINTN)mApHltLoopCode, TopOfStack, (UINTN)&mNumberToFinish);\r
529a5a86
MK
499}\r
500\r
501/**\r
502 Prepares startup vector for APs.\r
503\r
504 This function prepares startup vector for APs.\r
505\r
506 @param WorkingBuffer The address of the work buffer.\r
507**/\r
508VOID\r
509PrepareApStartupVector (\r
510 EFI_PHYSICAL_ADDRESS WorkingBuffer\r
511 )\r
512{\r
513 EFI_PHYSICAL_ADDRESS StartupVector;\r
514 MP_ASSEMBLY_ADDRESS_MAP AddressMap;\r
515\r
516 //\r
517 // Get the address map of startup code for AP,\r
518 // including code size, and offset of long jump instructions to redirect.\r
519 //\r
520 ZeroMem (&AddressMap, sizeof (AddressMap));\r
521 AsmGetAddressMap (&AddressMap);\r
522\r
523 StartupVector = WorkingBuffer;\r
524\r
525 //\r
526 // Copy AP startup code to startup vector, and then redirect the long jump\r
527 // instructions for mode switching.\r
528 //\r
529 CopyMem ((VOID *) (UINTN) StartupVector, AddressMap.RendezvousFunnelAddress, AddressMap.Size);\r
530 *(UINT32 *) (UINTN) (StartupVector + AddressMap.FlatJumpOffset + 3) = (UINT32) (StartupVector + AddressMap.PModeEntryOffset);\r
531 if (AddressMap.LongJumpOffset != 0) {\r
532 *(UINT32 *) (UINTN) (StartupVector + AddressMap.LongJumpOffset + 2) = (UINT32) (StartupVector + AddressMap.LModeEntryOffset);\r
533 }\r
534\r
535 //\r
536 // Get the start address of exchange data between BSP and AP.\r
537 //\r
538 mExchangeInfo = (MP_CPU_EXCHANGE_INFO *) (UINTN) (StartupVector + AddressMap.Size);\r
539 ZeroMem ((VOID *) mExchangeInfo, sizeof (MP_CPU_EXCHANGE_INFO));\r
540\r
541 CopyMem ((VOID *) (UINTN) &mExchangeInfo->GdtrProfile, (VOID *) (UINTN) mAcpiCpuData.GdtrProfile, sizeof (IA32_DESCRIPTOR));\r
542 CopyMem ((VOID *) (UINTN) &mExchangeInfo->IdtrProfile, (VOID *) (UINTN) mAcpiCpuData.IdtrProfile, sizeof (IA32_DESCRIPTOR));\r
543\r
529a5a86
MK
544 mExchangeInfo->StackStart = (VOID *) (UINTN) mAcpiCpuData.StackAddress;\r
545 mExchangeInfo->StackSize = mAcpiCpuData.StackSize;\r
546 mExchangeInfo->BufferStart = (UINT32) StartupVector;\r
547 mExchangeInfo->Cr3 = (UINT32) (AsmReadCr3 ());\r
e21e355e 548 mExchangeInfo->InitializeFloatingPointUnitsAddress = (UINTN)InitializeFloatingPointUnits;\r
529a5a86
MK
549}\r
550\r
551/**\r
552 The function is invoked before SMBASE relocation in S3 path to restores CPU status.\r
553\r
554 The function is invoked before SMBASE relocation in S3 path. It does first time microcode load\r
555 and restores MTRRs for both BSP and APs.\r
556\r
557**/\r
558VOID\r
94744aa2 559InitializeCpuBeforeRebase (\r
529a5a86
MK
560 VOID\r
561 )\r
562{\r
529a5a86
MK
563 LoadMtrrData (mAcpiCpuData.MtrrTable);\r
564\r
93324390 565 SetRegister (TRUE);\r
529a5a86
MK
566\r
567 ProgramVirtualWireMode ();\r
568\r
569 PrepareApStartupVector (mAcpiCpuData.StartupVector);\r
570\r
571 mNumberToFinish = mAcpiCpuData.NumberOfCpus - 1;\r
94744aa2 572 mExchangeInfo->ApFunction = (VOID *) (UINTN) InitializeAp;\r
7677b4db
ED
573\r
574 //\r
575 // Execute code for before SmmBaseReloc. Note: This flag is maintained across S3 boots.\r
576 //\r
577 mInitApsAfterSmmBaseReloc = FALSE;\r
529a5a86
MK
578\r
579 //\r
580 // Send INIT IPI - SIPI to all APs\r
581 //\r
582 SendInitSipiSipiAllExcludingSelf ((UINT32)mAcpiCpuData.StartupVector);\r
583\r
584 while (mNumberToFinish > 0) {\r
585 CpuPause ();\r
586 }\r
587}\r
588\r
589/**\r
590 The function is invoked after SMBASE relocation in S3 path to restores CPU status.\r
591\r
592 The function is invoked after SMBASE relocation in S3 path. It restores configuration according to\r
593 data saved by normal boot path for both BSP and APs.\r
594\r
595**/\r
596VOID\r
94744aa2 597InitializeCpuAfterRebase (\r
529a5a86
MK
598 VOID\r
599 )\r
600{\r
529a5a86 601 mNumberToFinish = mAcpiCpuData.NumberOfCpus - 1;\r
529a5a86
MK
602\r
603 //\r
93324390 604 // Signal that SMM base relocation is complete and to continue initialization for all APs.\r
529a5a86 605 //\r
7677b4db 606 mInitApsAfterSmmBaseReloc = TRUE;\r
529a5a86 607\r
93324390
ED
608 //\r
609 // Must begin set register after all APs have continue their initialization.\r
610 // This is a requirement to support semaphore mechanism in register table.\r
611 // Because if semaphore's dependence type is package type, semaphore will wait\r
612 // for all Aps in one package finishing their tasks before set next register\r
613 // for all APs. If the Aps not begin its task during BSP doing its task, the\r
614 // BSP thread will hang because it is waiting for other Aps in the same\r
615 // package finishing their task.\r
616 //\r
617 SetRegister (FALSE);\r
618\r
529a5a86
MK
619 while (mNumberToFinish > 0) {\r
620 CpuPause ();\r
621 }\r
622}\r
0bdc9e75
SZ
623\r
624/**\r
625 Restore SMM Configuration in S3 boot path.\r
626\r
627**/\r
628VOID\r
629RestoreSmmConfigurationInS3 (\r
630 VOID\r
631 )\r
632{\r
b10d5ddc
SZ
633 if (!mAcpiS3Enable) {\r
634 return;\r
635 }\r
636\r
0bdc9e75
SZ
637 //\r
638 // Restore SMM Configuration in S3 boot path.\r
639 //\r
640 if (mRestoreSmmConfigurationInS3) {\r
641 //\r
642 // Need make sure gSmst is correct because below function may use them.\r
643 //\r
644 gSmst->SmmStartupThisAp = gSmmCpuPrivate->SmmCoreEntryContext.SmmStartupThisAp;\r
645 gSmst->CurrentlyExecutingCpu = gSmmCpuPrivate->SmmCoreEntryContext.CurrentlyExecutingCpu;\r
646 gSmst->NumberOfCpus = gSmmCpuPrivate->SmmCoreEntryContext.NumberOfCpus;\r
647 gSmst->CpuSaveStateSize = gSmmCpuPrivate->SmmCoreEntryContext.CpuSaveStateSize;\r
648 gSmst->CpuSaveState = gSmmCpuPrivate->SmmCoreEntryContext.CpuSaveState;\r
649\r
650 //\r
651 // Configure SMM Code Access Check feature if available.\r
652 //\r
653 ConfigSmmCodeAccessCheck ();\r
654\r
655 SmmCpuFeaturesCompleteSmmReadyToLock ();\r
656\r
657 mRestoreSmmConfigurationInS3 = FALSE;\r
658 }\r
659}\r
660\r
661/**\r
662 Perform SMM initialization for all processors in the S3 boot path.\r
663\r
664 For a native platform, MP initialization in the S3 boot path is also performed in this function.\r
665**/\r
666VOID\r
667EFIAPI\r
668SmmRestoreCpu (\r
669 VOID\r
670 )\r
671{\r
672 SMM_S3_RESUME_STATE *SmmS3ResumeState;\r
673 IA32_DESCRIPTOR Ia32Idtr;\r
674 IA32_DESCRIPTOR X64Idtr;\r
675 IA32_IDT_GATE_DESCRIPTOR IdtEntryTable[EXCEPTION_VECTOR_NUMBER];\r
676 EFI_STATUS Status;\r
677\r
678 DEBUG ((EFI_D_INFO, "SmmRestoreCpu()\n"));\r
679\r
680 mSmmS3Flag = TRUE;\r
681\r
0bdc9e75
SZ
682 //\r
683 // See if there is enough context to resume PEI Phase\r
684 //\r
685 if (mSmmS3ResumeState == NULL) {\r
686 DEBUG ((EFI_D_ERROR, "No context to return to PEI Phase\n"));\r
687 CpuDeadLoop ();\r
688 }\r
689\r
690 SmmS3ResumeState = mSmmS3ResumeState;\r
691 ASSERT (SmmS3ResumeState != NULL);\r
692\r
693 if (SmmS3ResumeState->Signature == SMM_S3_RESUME_SMM_64) {\r
694 //\r
695 // Save the IA32 IDT Descriptor\r
696 //\r
697 AsmReadIdtr ((IA32_DESCRIPTOR *) &Ia32Idtr);\r
698\r
699 //\r
700 // Setup X64 IDT table\r
701 //\r
702 ZeroMem (IdtEntryTable, sizeof (IA32_IDT_GATE_DESCRIPTOR) * 32);\r
703 X64Idtr.Base = (UINTN) IdtEntryTable;\r
704 X64Idtr.Limit = (UINT16) (sizeof (IA32_IDT_GATE_DESCRIPTOR) * 32 - 1);\r
705 AsmWriteIdtr ((IA32_DESCRIPTOR *) &X64Idtr);\r
706\r
707 //\r
708 // Setup the default exception handler\r
709 //\r
710 Status = InitializeCpuExceptionHandlers (NULL);\r
711 ASSERT_EFI_ERROR (Status);\r
712\r
713 //\r
714 // Initialize Debug Agent to support source level debug\r
715 //\r
716 InitializeDebugAgent (DEBUG_AGENT_INIT_THUNK_PEI_IA32TOX64, (VOID *)&Ia32Idtr, NULL);\r
717 }\r
718\r
719 //\r
720 // Skip initialization if mAcpiCpuData is not valid\r
721 //\r
722 if (mAcpiCpuData.NumberOfCpus > 0) {\r
723 //\r
724 // First time microcode load and restore MTRRs\r
725 //\r
94744aa2 726 InitializeCpuBeforeRebase ();\r
0bdc9e75
SZ
727 }\r
728\r
729 //\r
730 // Restore SMBASE for BSP and all APs\r
731 //\r
732 SmmRelocateBases ();\r
733\r
734 //\r
735 // Skip initialization if mAcpiCpuData is not valid\r
736 //\r
737 if (mAcpiCpuData.NumberOfCpus > 0) {\r
738 //\r
739 // Restore MSRs for BSP and all APs\r
740 //\r
94744aa2 741 InitializeCpuAfterRebase ();\r
0bdc9e75
SZ
742 }\r
743\r
744 //\r
745 // Set a flag to restore SMM configuration in S3 path.\r
746 //\r
747 mRestoreSmmConfigurationInS3 = TRUE;\r
748\r
749 DEBUG (( EFI_D_INFO, "SMM S3 Return CS = %x\n", SmmS3ResumeState->ReturnCs));\r
750 DEBUG (( EFI_D_INFO, "SMM S3 Return Entry Point = %x\n", SmmS3ResumeState->ReturnEntryPoint));\r
751 DEBUG (( EFI_D_INFO, "SMM S3 Return Context1 = %x\n", SmmS3ResumeState->ReturnContext1));\r
752 DEBUG (( EFI_D_INFO, "SMM S3 Return Context2 = %x\n", SmmS3ResumeState->ReturnContext2));\r
753 DEBUG (( EFI_D_INFO, "SMM S3 Return Stack Pointer = %x\n", SmmS3ResumeState->ReturnStackPointer));\r
754\r
755 //\r
756 // If SMM is in 32-bit mode, then use SwitchStack() to resume PEI Phase\r
757 //\r
758 if (SmmS3ResumeState->Signature == SMM_S3_RESUME_SMM_32) {\r
759 DEBUG ((EFI_D_INFO, "Call SwitchStack() to return to S3 Resume in PEI Phase\n"));\r
760\r
761 SwitchStack (\r
762 (SWITCH_STACK_ENTRY_POINT)(UINTN)SmmS3ResumeState->ReturnEntryPoint,\r
763 (VOID *)(UINTN)SmmS3ResumeState->ReturnContext1,\r
764 (VOID *)(UINTN)SmmS3ResumeState->ReturnContext2,\r
765 (VOID *)(UINTN)SmmS3ResumeState->ReturnStackPointer\r
766 );\r
767 }\r
768\r
769 //\r
770 // If SMM is in 64-bit mode, then use AsmDisablePaging64() to resume PEI Phase\r
771 //\r
772 if (SmmS3ResumeState->Signature == SMM_S3_RESUME_SMM_64) {\r
773 DEBUG ((EFI_D_INFO, "Call AsmDisablePaging64() to return to S3 Resume in PEI Phase\n"));\r
774 //\r
775 // Disable interrupt of Debug timer, since new IDT table is for IA32 and will not work in long mode.\r
776 //\r
777 SaveAndSetDebugTimerInterrupt (FALSE);\r
778 //\r
779 // Restore IA32 IDT table\r
780 //\r
781 AsmWriteIdtr ((IA32_DESCRIPTOR *) &Ia32Idtr);\r
782 AsmDisablePaging64 (\r
783 SmmS3ResumeState->ReturnCs,\r
784 (UINT32)SmmS3ResumeState->ReturnEntryPoint,\r
785 (UINT32)SmmS3ResumeState->ReturnContext1,\r
786 (UINT32)SmmS3ResumeState->ReturnContext2,\r
787 (UINT32)SmmS3ResumeState->ReturnStackPointer\r
788 );\r
789 }\r
790\r
791 //\r
792 // Can not resume PEI Phase\r
793 //\r
794 DEBUG ((EFI_D_ERROR, "No context to return to PEI Phase\n"));\r
795 CpuDeadLoop ();\r
796}\r
797\r
798/**\r
799 Initialize SMM S3 resume state structure used during S3 Resume.\r
800\r
801 @param[in] Cr3 The base address of the page tables to use in SMM.\r
802\r
803**/\r
804VOID\r
805InitSmmS3ResumeState (\r
806 IN UINT32 Cr3\r
807 )\r
808{\r
809 VOID *GuidHob;\r
810 EFI_SMRAM_DESCRIPTOR *SmramDescriptor;\r
811 SMM_S3_RESUME_STATE *SmmS3ResumeState;\r
4a0f88dd
JF
812 EFI_PHYSICAL_ADDRESS Address;\r
813 EFI_STATUS Status;\r
0bdc9e75 814\r
b10d5ddc
SZ
815 if (!mAcpiS3Enable) {\r
816 return;\r
817 }\r
818\r
0bdc9e75 819 GuidHob = GetFirstGuidHob (&gEfiAcpiVariableGuid);\r
a95c9cfd
JW
820 if (GuidHob == NULL) {\r
821 DEBUG ((\r
822 DEBUG_ERROR,\r
823 "ERROR:%a(): HOB(gEfiAcpiVariableGuid=%g) needed by S3 resume doesn't exist!\n",\r
824 __FUNCTION__,\r
825 &gEfiAcpiVariableGuid\r
826 ));\r
827 CpuDeadLoop ();\r
828 } else {\r
0bdc9e75
SZ
829 SmramDescriptor = (EFI_SMRAM_DESCRIPTOR *) GET_GUID_HOB_DATA (GuidHob);\r
830\r
831 DEBUG ((EFI_D_INFO, "SMM S3 SMRAM Structure = %x\n", SmramDescriptor));\r
832 DEBUG ((EFI_D_INFO, "SMM S3 Structure = %x\n", SmramDescriptor->CpuStart));\r
833\r
834 SmmS3ResumeState = (SMM_S3_RESUME_STATE *)(UINTN)SmramDescriptor->CpuStart;\r
835 ZeroMem (SmmS3ResumeState, sizeof (SMM_S3_RESUME_STATE));\r
836\r
837 mSmmS3ResumeState = SmmS3ResumeState;\r
838 SmmS3ResumeState->Smst = (EFI_PHYSICAL_ADDRESS)(UINTN)gSmst;\r
839\r
840 SmmS3ResumeState->SmmS3ResumeEntryPoint = (EFI_PHYSICAL_ADDRESS)(UINTN)SmmRestoreCpu;\r
841\r
842 SmmS3ResumeState->SmmS3StackSize = SIZE_32KB;\r
843 SmmS3ResumeState->SmmS3StackBase = (EFI_PHYSICAL_ADDRESS)(UINTN)AllocatePages (EFI_SIZE_TO_PAGES ((UINTN)SmmS3ResumeState->SmmS3StackSize));\r
844 if (SmmS3ResumeState->SmmS3StackBase == 0) {\r
845 SmmS3ResumeState->SmmS3StackSize = 0;\r
846 }\r
847\r
f0053e83 848 SmmS3ResumeState->SmmS3Cr0 = mSmmCr0;\r
0bdc9e75 849 SmmS3ResumeState->SmmS3Cr3 = Cr3;\r
351b49c1 850 SmmS3ResumeState->SmmS3Cr4 = mSmmCr4;\r
0bdc9e75
SZ
851\r
852 if (sizeof (UINTN) == sizeof (UINT64)) {\r
853 SmmS3ResumeState->Signature = SMM_S3_RESUME_SMM_64;\r
854 }\r
855 if (sizeof (UINTN) == sizeof (UINT32)) {\r
856 SmmS3ResumeState->Signature = SMM_S3_RESUME_SMM_32;\r
857 }\r
0bdc9e75 858\r
16d84657
JW
859 //\r
860 // Patch SmmS3ResumeState->SmmS3Cr3\r
861 //\r
862 InitSmmS3Cr3 ();\r
863 }\r
4a0f88dd
JF
864\r
865 //\r
866 // Allocate safe memory in ACPI NVS for AP to execute hlt loop in\r
867 // protected mode on S3 path\r
868 //\r
869 Address = BASE_4GB - 1;\r
870 Status = gBS->AllocatePages (\r
871 AllocateMaxAddress,\r
872 EfiACPIMemoryNVS,\r
873 EFI_SIZE_TO_PAGES (sizeof (mApHltLoopCodeTemplate)),\r
874 &Address\r
875 );\r
876 ASSERT_EFI_ERROR (Status);\r
877 mApHltLoopCode = (UINT8 *) (UINTN) Address;\r
0bdc9e75
SZ
878}\r
879\r
880/**\r
881 Copy register table from ACPI NVS memory into SMRAM.\r
882\r
883 @param[in] DestinationRegisterTableList Points to destination register table.\r
884 @param[in] SourceRegisterTableList Points to source register table.\r
885 @param[in] NumberOfCpus Number of CPUs.\r
886\r
887**/\r
888VOID\r
889CopyRegisterTable (\r
890 IN CPU_REGISTER_TABLE *DestinationRegisterTableList,\r
891 IN CPU_REGISTER_TABLE *SourceRegisterTableList,\r
892 IN UINT32 NumberOfCpus\r
893 )\r
894{\r
895 UINTN Index;\r
0bdc9e75
SZ
896 CPU_REGISTER_TABLE_ENTRY *RegisterTableEntry;\r
897\r
898 CopyMem (DestinationRegisterTableList, SourceRegisterTableList, NumberOfCpus * sizeof (CPU_REGISTER_TABLE));\r
899 for (Index = 0; Index < NumberOfCpus; Index++) {\r
30d995ee
JF
900 if (DestinationRegisterTableList[Index].AllocatedSize != 0) {\r
901 RegisterTableEntry = AllocateCopyPool (\r
902 DestinationRegisterTableList[Index].AllocatedSize,\r
903 (VOID *)(UINTN)SourceRegisterTableList[Index].RegisterTableEntry\r
904 );\r
905 ASSERT (RegisterTableEntry != NULL);\r
906 DestinationRegisterTableList[Index].RegisterTableEntry = (EFI_PHYSICAL_ADDRESS)(UINTN)RegisterTableEntry;\r
0bdc9e75
SZ
907 }\r
908 }\r
909}\r
910\r
911/**\r
912 Get ACPI CPU data.\r
913\r
914**/\r
915VOID\r
916GetAcpiCpuData (\r
917 VOID\r
918 )\r
919{\r
920 ACPI_CPU_DATA *AcpiCpuData;\r
921 IA32_DESCRIPTOR *Gdtr;\r
922 IA32_DESCRIPTOR *Idtr;\r
293f8766
ED
923 VOID *GdtForAp;\r
924 VOID *IdtForAp;\r
925 VOID *MachineCheckHandlerForAp;\r
93324390 926 CPU_STATUS_INFORMATION *CpuStatus;\r
0bdc9e75 927\r
b10d5ddc
SZ
928 if (!mAcpiS3Enable) {\r
929 return;\r
930 }\r
931\r
0bdc9e75
SZ
932 //\r
933 // Prevent use of mAcpiCpuData by initialize NumberOfCpus to 0\r
934 //\r
935 mAcpiCpuData.NumberOfCpus = 0;\r
936\r
937 //\r
938 // If PcdCpuS3DataAddress was never set, then do not copy CPU S3 Data into SMRAM\r
939 //\r
940 AcpiCpuData = (ACPI_CPU_DATA *)(UINTN)PcdGet64 (PcdCpuS3DataAddress);\r
941 if (AcpiCpuData == 0) {\r
942 return;\r
943 }\r
944\r
945 //\r
946 // For a native platform, copy the CPU S3 data into SMRAM for use on CPU S3 Resume.\r
947 //\r
948 CopyMem (&mAcpiCpuData, AcpiCpuData, sizeof (mAcpiCpuData));\r
949\r
950 mAcpiCpuData.MtrrTable = (EFI_PHYSICAL_ADDRESS)(UINTN)AllocatePool (sizeof (MTRR_SETTINGS));\r
951 ASSERT (mAcpiCpuData.MtrrTable != 0);\r
952\r
953 CopyMem ((VOID *)(UINTN)mAcpiCpuData.MtrrTable, (VOID *)(UINTN)AcpiCpuData->MtrrTable, sizeof (MTRR_SETTINGS));\r
954\r
955 mAcpiCpuData.GdtrProfile = (EFI_PHYSICAL_ADDRESS)(UINTN)AllocatePool (sizeof (IA32_DESCRIPTOR));\r
956 ASSERT (mAcpiCpuData.GdtrProfile != 0);\r
957\r
958 CopyMem ((VOID *)(UINTN)mAcpiCpuData.GdtrProfile, (VOID *)(UINTN)AcpiCpuData->GdtrProfile, sizeof (IA32_DESCRIPTOR));\r
959\r
960 mAcpiCpuData.IdtrProfile = (EFI_PHYSICAL_ADDRESS)(UINTN)AllocatePool (sizeof (IA32_DESCRIPTOR));\r
961 ASSERT (mAcpiCpuData.IdtrProfile != 0);\r
962\r
963 CopyMem ((VOID *)(UINTN)mAcpiCpuData.IdtrProfile, (VOID *)(UINTN)AcpiCpuData->IdtrProfile, sizeof (IA32_DESCRIPTOR));\r
964\r
965 mAcpiCpuData.PreSmmInitRegisterTable = (EFI_PHYSICAL_ADDRESS)(UINTN)AllocatePool (mAcpiCpuData.NumberOfCpus * sizeof (CPU_REGISTER_TABLE));\r
966 ASSERT (mAcpiCpuData.PreSmmInitRegisterTable != 0);\r
967\r
968 CopyRegisterTable (\r
969 (CPU_REGISTER_TABLE *)(UINTN)mAcpiCpuData.PreSmmInitRegisterTable,\r
970 (CPU_REGISTER_TABLE *)(UINTN)AcpiCpuData->PreSmmInitRegisterTable,\r
971 mAcpiCpuData.NumberOfCpus\r
972 );\r
973\r
974 mAcpiCpuData.RegisterTable = (EFI_PHYSICAL_ADDRESS)(UINTN)AllocatePool (mAcpiCpuData.NumberOfCpus * sizeof (CPU_REGISTER_TABLE));\r
975 ASSERT (mAcpiCpuData.RegisterTable != 0);\r
976\r
977 CopyRegisterTable (\r
978 (CPU_REGISTER_TABLE *)(UINTN)mAcpiCpuData.RegisterTable,\r
979 (CPU_REGISTER_TABLE *)(UINTN)AcpiCpuData->RegisterTable,\r
980 mAcpiCpuData.NumberOfCpus\r
981 );\r
982\r
983 //\r
984 // Copy AP's GDT, IDT and Machine Check handler into SMRAM.\r
985 //\r
986 Gdtr = (IA32_DESCRIPTOR *)(UINTN)mAcpiCpuData.GdtrProfile;\r
987 Idtr = (IA32_DESCRIPTOR *)(UINTN)mAcpiCpuData.IdtrProfile;\r
988\r
293f8766
ED
989 GdtForAp = AllocatePool ((Gdtr->Limit + 1) + (Idtr->Limit + 1) + mAcpiCpuData.ApMachineCheckHandlerSize);\r
990 ASSERT (GdtForAp != NULL);\r
991 IdtForAp = (VOID *) ((UINTN)GdtForAp + (Gdtr->Limit + 1));\r
992 MachineCheckHandlerForAp = (VOID *) ((UINTN)IdtForAp + (Idtr->Limit + 1));\r
993\r
994 CopyMem (GdtForAp, (VOID *)Gdtr->Base, Gdtr->Limit + 1);\r
995 CopyMem (IdtForAp, (VOID *)Idtr->Base, Idtr->Limit + 1);\r
996 CopyMem (MachineCheckHandlerForAp, (VOID *)(UINTN)mAcpiCpuData.ApMachineCheckHandlerBase, mAcpiCpuData.ApMachineCheckHandlerSize);\r
0bdc9e75 997\r
293f8766
ED
998 Gdtr->Base = (UINTN)GdtForAp;\r
999 Idtr->Base = (UINTN)IdtForAp;\r
1000 mAcpiCpuData.ApMachineCheckHandlerBase = (EFI_PHYSICAL_ADDRESS)(UINTN)MachineCheckHandlerForAp;\r
93324390
ED
1001\r
1002 CpuStatus = &mAcpiCpuData.CpuStatus;\r
1003 CopyMem (CpuStatus, &AcpiCpuData->CpuStatus, sizeof (CPU_STATUS_INFORMATION));\r
1004 if (AcpiCpuData->CpuStatus.ValidCoreCountPerPackage != 0) {\r
1005 CpuStatus->ValidCoreCountPerPackage = (EFI_PHYSICAL_ADDRESS)(UINTN)AllocateCopyPool (\r
1006 sizeof (UINT32) * CpuStatus->PackageCount,\r
1007 (UINT32 *)(UINTN)AcpiCpuData->CpuStatus.ValidCoreCountPerPackage\r
1008 );\r
1009 ASSERT (CpuStatus->ValidCoreCountPerPackage != 0);\r
1010 }\r
1011 if (AcpiCpuData->ApLocation != 0) {\r
1012 mAcpiCpuData.ApLocation = (EFI_PHYSICAL_ADDRESS)(UINTN)AllocateCopyPool (\r
1013 mAcpiCpuData.NumberOfCpus * sizeof (EFI_CPU_PHYSICAL_LOCATION),\r
1014 (EFI_CPU_PHYSICAL_LOCATION *)(UINTN)AcpiCpuData->ApLocation\r
1015 );\r
1016 ASSERT (mAcpiCpuData.ApLocation != 0);\r
1017 }\r
1018 if (CpuStatus->PackageCount != 0) {\r
9bae7811
ED
1019 mCpuFlags.CoreSemaphoreCount = AllocateZeroPool (\r
1020 sizeof (UINT32) * CpuStatus->PackageCount *\r
1021 CpuStatus->MaxCoreCount * CpuStatus->MaxThreadCount\r
1022 );\r
1023 ASSERT (mCpuFlags.CoreSemaphoreCount != NULL);\r
1024 mCpuFlags.PackageSemaphoreCount = AllocateZeroPool (\r
1025 sizeof (UINT32) * CpuStatus->PackageCount *\r
1026 CpuStatus->MaxCoreCount * CpuStatus->MaxThreadCount\r
1027 );\r
1028 ASSERT (mCpuFlags.PackageSemaphoreCount != NULL);\r
93324390
ED
1029 }\r
1030 InitializeSpinLock((SPIN_LOCK*) &mCpuFlags.MemoryMappedLock);\r
1031 InitializeSpinLock((SPIN_LOCK*) &mCpuFlags.ConsoleLogLock);\r
0bdc9e75 1032}\r
b10d5ddc
SZ
1033\r
1034/**\r
1035 Get ACPI S3 enable flag.\r
1036\r
1037**/\r
1038VOID\r
1039GetAcpiS3EnableFlag (\r
1040 VOID\r
1041 )\r
1042{\r
1043 mAcpiS3Enable = PcdGetBool (PcdAcpiS3Enable);\r
1044}\r