]> git.proxmox.com Git - mirror_edk2.git/blobdiff - EdkModulePkg/Library/EdkDxeSalLib/Ipf/EsalServiceLib.c
Further check-in to smooth Intel IPF compiler building.
[mirror_edk2.git] / EdkModulePkg / Library / EdkDxeSalLib / Ipf / EsalServiceLib.c
index 83b6aa2c4cfb106e4659c386471069abafd1c603..2b4d2972213d6e8562cbc9ffa001605979609c04 100644 (file)
@@ -1,13 +1,13 @@
 /*++\r
 \r
-Copyright (c) 2006, 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
+Copyright (c) 2006, 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
@@ -19,22 +19,75 @@ Abstract:
 \r
 #include <Ipf/IpfDefines.h>\r
 \r
+EXTENDED_SAL_BOOT_SERVICE_PROTOCOL  *mEsalBootService = NULL;\r
+EFI_PLABEL                          mPlabel;\r
 \r
-STATIC EXTENDED_SAL_BOOT_SERVICE_PROTOCOL *mEsalBootService;\r
-\r
+STATIC\r
 EFI_STATUS\r
 EFIAPI\r
-DxeSalLibConstruct (\r
-  IN EFI_HANDLE        ImageHandle,\r
-  IN EFI_SYSTEM_TABLE  *SystemTable\r
+DxeSalLibInitialize (\r
+  VOID\r
   )\r
 {\r
-  EFI_STATUS Status;\r
+  EFI_PLABEL  *Plabel;\r
+  EFI_STATUS  Status;\r
+\r
+  if (mEsalBootService != NULL) {\r
+    return EFI_SUCCESS;\r
+  }\r
+\r
+  //\r
+  // The protocol contains a function pointer, which is an indirect procedure call.\r
+  // An indirect procedure call goes through a plabel, and pointer to a function is\r
+  // a pointer to a plabel. To implement indirect procedure calls that can work in\r
+  // both physical and virtual mode, two plabels are required (one physical and one\r
+  // virtual). So lets grap the physical PLABEL for the EsalEntryPoint and store it\r
+  // away. We cache it in a module global, so we can register the vitrual version.\r
+  //\r
+  Status = gBS->LocateProtocol (&gEfiExtendedSalBootServiceProtocolGuid, NULL, (VOID **) &mEsalBootService);\r
+  if (EFI_ERROR (Status)) {\r
+    mEsalBootService = NULL;\r
+    return EFI_SUCCESS;\r
+  }\r
+\r
+  Plabel              = (EFI_PLABEL *) (UINTN) mEsalBootService->ExtendedSalProc;\r
+\r
+  mPlabel.EntryPoint  = Plabel->EntryPoint;\r
+  mPlabel.GP          = Plabel->GP;\r
+  SetEsalPhysicalEntryPoint (mPlabel.EntryPoint, mPlabel.GP);\r
+\r
+  return EFI_SUCCESS;\r
+}\r
+\r
+STATIC\r
+VOID\r
+EFIAPI\r
+DxeSalVirtualNotifyEvent (\r
+  IN EFI_EVENT        Event,\r
+  IN VOID             *Context\r
+  )\r
+/*++\r
+\r
+Routine Description:\r
+\r
+  Fixup virtual address pointer of label.\r
+\r
+Arguments:\r
+\r
+  Event   - The Event that is being processed\r
+\r
+  Context - Event Context\r
+\r
+Returns:\r
+\r
+  None\r
 \r
-  Status = gBS->LocateProtocol (&gEfiExtendedSalBootServiceProtocolGuid, NULL, &mEsalBootService);\r
-  ASSERT_EFI_ERROR (Status);\r
+--*/\r
+{\r
+  EfiConvertPointer (0x0, (VOID **) &mPlabel.EntryPoint);\r
+  EfiConvertPointer (EFI_IPF_GP_POINTER, (VOID **) &mPlabel.GP);\r
 \r
-  return Status;\r
+  SetEsalVirtualEntryPoint (mPlabel.EntryPoint, mPlabel.GP);\r
 }\r
 \r
 EFI_STATUS\r
@@ -54,15 +107,16 @@ Routine Description:
 \r
 Arguments:\r
   FunctionId    - ID of function to register\r
-  ClassGuid     - GUID of function class \r
+  ClassGuid     - GUID of function class\r
   Function      - Function to register under ClassGuid/FunctionId pair\r
   ModuleGlobal  - Module global for Function.\r
 \r
-Returns: \r
+Returns:\r
   EFI_SUCCESS - If ClassGuid/FunctionId Function was registered.\r
 \r
 --*/\r
 {\r
+  DxeSalLibInitialize ();\r
   return mEsalBootService->AddExtendedSalProc (\r
                             mEsalBootService,\r
                             ClassGuid,\r
@@ -87,12 +141,12 @@ Routine Description:
   This function is boot service only!\r
 \r
 Arguments:\r
-  ClassGuid     - GUID of function class \r
+  ClassGuid     - GUID of function class\r
   ModuleGlobal  - Module global for Function.\r
-  ...           - SAL_INTERNAL_EXTENDED_SAL_PROC and FunctionId pairs. NULL \r
+  ...           - SAL_INTERNAL_EXTENDED_SAL_PROC and FunctionId pairs. NULL\r
                   indicates the end of the list.\r
 \r
-Returns: \r
+Returns:\r
   EFI_SUCCESS - All members of ClassGuid registered\r
 \r
 --*/\r
@@ -147,7 +201,7 @@ EfiCallEsalService (
 \r
 Routine Description:\r
 \r
-  Call module that is not linked direclty to this module. This code is IP \r
+  Call module that is not linked direclty to this module. This code is IP\r
   relative and hides the binding issues of virtual or physical calling. The\r
   function that gets dispatched has extra arguments that include the registered\r
   module global and a boolean flag to indicate if the system is in virutal mode.\r
@@ -163,7 +217,7 @@ Arguments:
   Arg7        - Argument 7 ClassGuid/FunctionId defined\r
   Arg8        - Argument 8 ClassGuid/FunctionId defined\r
 \r
-Returns: \r
+Returns:\r
   Status of ClassGuid/FuncitonId\r
 \r
 --*/\r
@@ -176,6 +230,27 @@ Returns:
     return ReturnReg;\r
   }\r
 \r
+  //\r
+  // Look at the physical mode ESAL entry point to determine of the ESAL entry point has been initialized\r
+  //\r
+  if (*(UINT64 *)ReturnReg.r9 == 0 && *(UINT64 *)(ReturnReg.r9 + 8) == 0) {\r
+    //\r
+    // Both the function ponter and the GP value are zero, so attempt to initialize the ESAL Entry Point\r
+    //\r
+    DxeSalLibInitialize ();\r
+    ReturnReg = GetEsalEntryPoint ();\r
+    if (ReturnReg.Status != EFI_SAL_SUCCESS) {\r
+      return ReturnReg;\r
+    }\r
+    if (*(UINT64 *)ReturnReg.r9 == 0 && *(UINT64 *)(ReturnReg.r9 + 8) == 0) {\r
+      //\r
+      // The ESAL Entry Point could not be initialized\r
+      //\r
+      ReturnReg.Status = EFI_SAL_ERROR;\r
+      return ReturnReg;\r
+    }\r
+  }\r
+\r
   if (ReturnReg.r11 & PSR_IT_MASK) {\r
     //\r
     // Virtual mode plabel to entry point\r