3 Copyright (c) 2006, Intel Corporation
4 All rights reserved. This program and the accompanying materials
5 are licensed and made available under the terms and conditions of the BSD License
6 which accompanies this distribution. The full text of the license may be found at
7 http://opensource.org/licenses/bsd-license.php
9 THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
10 WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
20 #include <Ipf/IpfDefines.h>
22 EXTENDED_SAL_BOOT_SERVICE_PROTOCOL
*mEsalBootService
= NULL
;
34 if (mEsalBootService
!= NULL
) {
39 // The protocol contains a function pointer, which is an indirect procedure call.
40 // An indirect procedure call goes through a plabel, and pointer to a function is
41 // a pointer to a plabel. To implement indirect procedure calls that can work in
42 // both physical and virtual mode, two plabels are required (one physical and one
43 // virtual). So lets grap the physical PLABEL for the EsalEntryPoint and store it
44 // away. We cache it in a module global, so we can register the vitrual version.
46 Status
= gBS
->LocateProtocol (&gEfiExtendedSalBootServiceProtocolGuid
, NULL
, &mEsalBootService
);
47 if (EFI_ERROR (Status
)) {
48 mEsalBootService
= NULL
;
52 Plabel
= (EFI_PLABEL
*) (UINTN
) mEsalBootService
->ExtendedSalProc
;
54 mPlabel
.EntryPoint
= Plabel
->EntryPoint
;
55 mPlabel
.GP
= Plabel
->GP
;
56 SetEsalPhysicalEntryPoint (mPlabel
.EntryPoint
, mPlabel
.GP
);
63 DxeSalVirtualNotifyEvent (
71 Fixup virtual address pointer of label.
75 Event - The Event that is being processed
77 Context - Event Context
85 EfiConvertPointer (0x0, (VOID
**) &mPlabel
.EntryPoint
);
86 EfiConvertPointer (EFI_IPF_GP_POINTER
, (VOID
**) &mPlabel
.GP
);
88 SetEsalVirtualEntryPoint (mPlabel
.EntryPoint
, mPlabel
.GP
);
93 RegisterEsalFunction (
95 IN EFI_GUID
*ClassGuid
,
96 IN SAL_INTERNAL_EXTENDED_SAL_PROC Function
,
103 Register ESAL Class Function and it's asociated global.
104 This function is boot service only!
107 FunctionId - ID of function to register
108 ClassGuid - GUID of function class
109 Function - Function to register under ClassGuid/FunctionId pair
110 ModuleGlobal - Module global for Function.
113 EFI_SUCCESS - If ClassGuid/FunctionId Function was registered.
117 DxeSalLibInitialize ();
118 return mEsalBootService
->AddExtendedSalProc (
130 IN EFI_GUID
*ClassGuid
,
131 IN VOID
*ModuleGlobal
,
138 Register ESAL Class and it's asociated global.
139 This function is boot service only!
142 ClassGuid - GUID of function class
143 ModuleGlobal - Module global for Function.
144 ... - SAL_INTERNAL_EXTENDED_SAL_PROC and FunctionId pairs. NULL
145 indicates the end of the list.
148 EFI_SUCCESS - All members of ClassGuid registered
154 SAL_INTERNAL_EXTENDED_SAL_PROC Function
;
156 EFI_HANDLE NewHandle
;
158 VA_START (Args
, ModuleGlobal
);
160 Status
= EFI_SUCCESS
;
161 while (!EFI_ERROR (Status
)) {
162 Function
= (SAL_INTERNAL_EXTENDED_SAL_PROC
) VA_ARG (Args
, SAL_INTERNAL_EXTENDED_SAL_PROC
);
163 if (Function
== NULL
) {
167 FunctionId
= VA_ARG (Args
, UINT64
);
169 Status
= RegisterEsalFunction (FunctionId
, ClassGuid
, Function
, ModuleGlobal
);
172 if (EFI_ERROR (Status
)) {
177 return gBS
->InstallProtocolInterface (
180 EFI_NATIVE_INTERFACE
,
188 IN EFI_GUID
*ClassGuid
,
189 IN UINT64 FunctionId
,
202 Call module that is not linked direclty to this module. This code is IP
203 relative and hides the binding issues of virtual or physical calling. The
204 function that gets dispatched has extra arguments that include the registered
205 module global and a boolean flag to indicate if the system is in virutal mode.
208 ClassGuid - GUID of function
209 FunctionId - Function in ClassGuid to call
210 Arg2 - Argument 2 ClassGuid/FunctionId defined
211 Arg3 - Argument 3 ClassGuid/FunctionId defined
212 Arg4 - Argument 4 ClassGuid/FunctionId defined
213 Arg5 - Argument 5 ClassGuid/FunctionId defined
214 Arg6 - Argument 6 ClassGuid/FunctionId defined
215 Arg7 - Argument 7 ClassGuid/FunctionId defined
216 Arg8 - Argument 8 ClassGuid/FunctionId defined
219 Status of ClassGuid/FuncitonId
223 SAL_RETURN_REGS ReturnReg
;
224 SAL_EXTENDED_SAL_PROC EsalProc
;
226 ReturnReg
= GetEsalEntryPoint ();
227 if (ReturnReg
.Status
!= EFI_SAL_SUCCESS
) {
232 // Look at the physical mode ESAL entry point to determine of the ESAL entry point has been initialized
234 if (*(UINT64
*)ReturnReg
.r9
== 0 && *(UINT64
*)(ReturnReg
.r9
+ 8) == 0) {
236 // Both the function ponter and the GP value are zero, so attempt to initialize the ESAL Entry Point
238 DxeSalLibInitialize ();
239 ReturnReg
= GetEsalEntryPoint ();
240 if (ReturnReg
.Status
!= EFI_SAL_SUCCESS
) {
243 if (*(UINT64
*)ReturnReg
.r9
== 0 && *(UINT64
*)(ReturnReg
.r9
+ 8) == 0) {
245 // The ESAL Entry Point could not be initialized
247 ReturnReg
.Status
= EFI_SAL_ERROR
;
252 if (ReturnReg
.r11
& PSR_IT_MASK
) {
254 // Virtual mode plabel to entry point
256 EsalProc
= (SAL_EXTENDED_SAL_PROC
) ReturnReg
.r10
;
259 // Physical mode plabel to entry point
261 EsalProc
= (SAL_EXTENDED_SAL_PROC
) ReturnReg
.r9
;