]> git.proxmox.com Git - mirror_edk2.git/blobdiff - EdkCompatibilityPkg/Foundation/Library/Pei/PeiLib/ia32/PeiServicePointer.c
Add in the 1st version of ECP.
[mirror_edk2.git] / EdkCompatibilityPkg / Foundation / Library / Pei / PeiLib / ia32 / PeiServicePointer.c
diff --git a/EdkCompatibilityPkg/Foundation/Library/Pei/PeiLib/ia32/PeiServicePointer.c b/EdkCompatibilityPkg/Foundation/Library/Pei/PeiLib/ia32/PeiServicePointer.c
new file mode 100644 (file)
index 0000000..54241b2
--- /dev/null
@@ -0,0 +1,152 @@
+/*++\r
+\r
+Copyright (c) 2004 - 2007, Intel Corporation                                                         \r
+All rights reserved. This program and the accompanying materials                          \r
+are licensed and made available under the terms and conditions of the BSD License         \r
+which accompanies this distribution.  The full text of the license may be found at        \r
+http://opensource.org/licenses/bsd-license.php                                            \r
+                                                                                          \r
+THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,                     \r
+WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.             \r
+\r
+Module Name:\r
+\r
+    PeiServicePointer.c\r
+\r
+Abstract:\r
+\r
+--*/\r
+\r
+#include "Tiano.h"\r
+#include "PeiApi.h"\r
+#include "PeiLib.h"\r
+\r
+#if (PI_SPECIFICATION_VERSION >= 0x00010000)\r
+\r
+#ifdef EFI_NT_EMULATOR\r
+EFI_PEI_SERVICES  **gPeiServices;\r
+#endif\r
+\r
+\r
+VOID\r
+SetPeiServicesTablePointer (\r
+  IN EFI_PEI_SERVICES  **PeiServices\r
+  )\r
+/*++\r
+\r
+Routine Description:\r
+\r
+  Save PeiService pointer so that it can be retrieved anywhere.\r
+\r
+Arguments:\r
+\r
+  PeiServices     - The direct pointer to PeiServiceTable.\r
+  PhyscialAddress - The physcial address of variable PeiServices.\r
+  \r
+Returns:\r
+  NONE\r
+  \r
+--*/          \r
+{\r
+\r
+#ifdef EFI_NT_EMULATOR\r
+\r
+  //\r
+  // For NT32, set EFI_PEI_SERVICES** to global variable.\r
+  //\r
+  gPeiServices = PeiServices;\r
+#else\r
+\r
+  //\r
+  // For X86 processor,the EFI_PEI_SERVICES** is stored in the \r
+  // 4 bytes immediately preceding the Interrupt Descriptor Table.\r
+  //\r
+  UINTN IdtBaseAddress;\r
+  IdtBaseAddress = (UINTN)ReadIdtBase();\r
+  *(UINTN*)(IdtBaseAddress - 4) = (UINTN)PeiServices;\r
+\r
+#endif  \r
+}\r
+\r
+\r
+EFI_PEI_SERVICES **\r
+GetPeiServicesTablePointer ( \r
+  VOID\r
+  )\r
+/*++\r
+\r
+Routine Description:\r
+\r
+  Get PeiService pointer.\r
+\r
+Arguments:\r
+\r
+  NONE.\r
+  \r
+Returns:\r
+  The direct pointer to PeiServiceTable.\r
+  \r
+--*/          \r
+{\r
+  EFI_PEI_SERVICES  **PeiServices;\r
+\r
+#ifdef EFI_NT_EMULATOR\r
+\r
+  //\r
+  // For NT32, set EFI_PEI_SERVICES** to global variable.\r
+  //\r
+  PeiServices = gPeiServices;\r
+#else\r
+\r
+  //\r
+  // For X86 processor,the EFI_PEI_SERVICES** is stored in the \r
+  // 4 bytes immediately preceding the Interrupt Descriptor Table.\r
+  //\r
+  UINTN IdtBaseAddress;\r
+  IdtBaseAddress = (UINTN)ReadIdtBase();\r
+  PeiServices = (EFI_PEI_SERVICES **)(UINTN)(*(UINTN*)(IdtBaseAddress - 4));\r
+#endif\r
+  return PeiServices;\r
+}\r
+\r
+\r
+VOID\r
+MigrateIdtTable (\r
+  IN EFI_PEI_SERVICES  **PeiServices\r
+  )\r
+/*++\r
+\r
+Routine Description:\r
+\r
+  Migrate IDT from CAR to real memory where preceded with 4 bytes for\r
+  storing PeiService pointer.\r
+\r
+Arguments:\r
+\r
+  PeiServices   - The direct pointer to PeiServiceTable.\r
+  \r
+Returns:\r
+\r
+  NONE.\r
+  \r
+--*/            \r
+{\r
+#ifndef EFI_NT_EMULATOR\r
+  UINT16          IdtEntrySize;\r
+  UINTN           OldIdtBase;\r
+  UINTN           Size;\r
+  VOID            *NewIdtBase;\r
+  EFI_STATUS      Status;\r
+  \r
+  IdtEntrySize    = ReadIdtLimit();\r
+  OldIdtBase      = ReadIdtBase();\r
+  Size = sizeof(PEI_IDT_TABLE) + (IdtEntrySize + 1);  \r
+  Status = (*PeiServices)->AllocatePool (PeiServices, Size, &NewIdtBase);\r
+  ASSERT_PEI_ERROR (PeiServices, Status);\r
+  (*PeiServices)->CopyMem ((VOID*)((UINTN)NewIdtBase + sizeof(PEI_IDT_TABLE)), (VOID*)OldIdtBase, (IdtEntrySize + 1));\r
+  SetIdtBase(((UINTN)NewIdtBase + sizeof(PEI_IDT_TABLE)), IdtEntrySize);\r
+  SetPeiServicesTablePointer(PeiServices);\r
+#endif\r
+}\r
+\r
+#endif\r