2 This module contains EBC support routines that are customized based on
3 the target x64 processor.
5 Copyright (c) 2006 - 2010, Intel Corporation. All rights reserved.<BR>
6 This program and the accompanying materials
7 are licensed and made available under the terms and conditions of the BSD License
8 which accompanies this distribution. The full text of the license may be found at
9 http://opensource.org/licenses/bsd-license.php
11 THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
12 WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
17 #include "EbcExecute.h"
20 // NOTE: This is the stack size allocated for the interpreter
21 // when it executes an EBC image. The requirements can change
22 // based on whether or not a debugger is present, and other
23 // platform-specific configurations.
25 #define VM_STACK_SIZE (1024 * 8)
26 #define EBC_THUNK_SIZE 64
28 #define STACK_REMAIN_SIZE (1024 * 4)
32 Pushes a 64 bit unsigned value to the VM stack.
34 @param VmPtr The pointer to current VM context.
35 @param Arg The value to be pushed.
45 // Advance the VM stack down, and then copy the argument to the stack.
48 VmPtr
->Gpr
[0] -= sizeof (UINT64
);
49 *(UINT64
*) VmPtr
->Gpr
[0] = Arg
;
55 Begin executing an EBC image. The address of the entry point is passed
56 in via a processor register, so we'll need to make a call to get the
59 This is a thunk function. Microsoft x64 compiler only provide fast_call
60 calling convention, so the first four arguments are passed by rcx, rdx,
61 r8, and r9, while other arguments are passed in stack.
63 @param Arg1 The 1st argument.
64 @param Arg2 The 2nd argument.
65 @param Arg3 The 3rd argument.
66 @param Arg4 The 4th argument.
67 @param Arg5 The 5th argument.
68 @param Arg6 The 6th argument.
69 @param Arg7 The 7th argument.
70 @param Arg8 The 8th argument.
71 @param Arg9 The 9th argument.
72 @param Arg10 The 10th argument.
73 @param Arg11 The 11th argument.
74 @param Arg12 The 12th argument.
75 @param Arg13 The 13th argument.
76 @param Arg14 The 14th argument.
77 @param Arg15 The 15th argument.
78 @param Arg16 The 16th argument.
80 @return The value returned by the EBC application we're going to run.
104 // Create a new VM context on the stack
106 VM_CONTEXT VmContext
;
112 // Get the EBC entry point from the processor register.
113 // Don't call any function before getting the EBC entry
114 // point because this will collab the return register.
116 Addr
= EbcLLGetEbcEntryPoint ();
119 // Now clear out our context
121 ZeroMem ((VOID
*) &VmContext
, sizeof (VM_CONTEXT
));
124 // Set the VM instruction pointer to the correct location in memory.
126 VmContext
.Ip
= (VMIP
) Addr
;
129 // Initialize the stack pointer for the EBC. Get the current system stack
130 // pointer and adjust it down by the max needed for the interpreter.
132 Addr
= EbcLLGetStackPointer ();
135 // Adjust the VM's stack pointer down.
138 Status
= GetEBCStack((EFI_HANDLE
)(UINTN
)-1, &VmContext
.StackPool
, &StackIndex
);
139 if (EFI_ERROR(Status
)) {
142 VmContext
.StackTop
= (UINT8
*)VmContext
.StackPool
+ (STACK_REMAIN_SIZE
);
143 VmContext
.Gpr
[0] = (UINT64
) ((UINT8
*)VmContext
.StackPool
+ STACK_POOL_SIZE
);
144 VmContext
.HighStackBottom
= (UINTN
) VmContext
.Gpr
[0];
145 VmContext
.Gpr
[0] -= sizeof (UINTN
);
148 // Align the stack on a natural boundary.
150 VmContext
.Gpr
[0] &= ~(sizeof (UINTN
) - 1);
153 // Put a magic value in the stack gap, then adjust down again.
155 *(UINTN
*) (UINTN
) (VmContext
.Gpr
[0]) = (UINTN
) VM_STACK_KEY_VALUE
;
156 VmContext
.StackMagicPtr
= (UINTN
*) (UINTN
) VmContext
.Gpr
[0];
159 // The stack upper to LowStackTop is belong to the VM.
161 VmContext
.LowStackTop
= (UINTN
) VmContext
.Gpr
[0];
164 // For the worst case, assume there are 4 arguments passed in registers, store
165 // them to VM's stack.
167 PushU64 (&VmContext
, (UINT64
) Arg16
);
168 PushU64 (&VmContext
, (UINT64
) Arg15
);
169 PushU64 (&VmContext
, (UINT64
) Arg14
);
170 PushU64 (&VmContext
, (UINT64
) Arg13
);
171 PushU64 (&VmContext
, (UINT64
) Arg12
);
172 PushU64 (&VmContext
, (UINT64
) Arg11
);
173 PushU64 (&VmContext
, (UINT64
) Arg10
);
174 PushU64 (&VmContext
, (UINT64
) Arg9
);
175 PushU64 (&VmContext
, (UINT64
) Arg8
);
176 PushU64 (&VmContext
, (UINT64
) Arg7
);
177 PushU64 (&VmContext
, (UINT64
) Arg6
);
178 PushU64 (&VmContext
, (UINT64
) Arg5
);
179 PushU64 (&VmContext
, (UINT64
) Arg4
);
180 PushU64 (&VmContext
, (UINT64
) Arg3
);
181 PushU64 (&VmContext
, (UINT64
) Arg2
);
182 PushU64 (&VmContext
, (UINT64
) Arg1
);
185 // Interpreter assumes 64-bit return address is pushed on the stack.
186 // The x64 does not do this so pad the stack accordingly.
188 PushU64 (&VmContext
, (UINT64
) 0);
189 PushU64 (&VmContext
, (UINT64
) 0x1234567887654321ULL
);
192 // For x64, this is where we say our return address is
194 VmContext
.StackRetAddr
= (UINT64
) VmContext
.Gpr
[0];
197 // We need to keep track of where the EBC stack starts. This way, if the EBC
198 // accesses any stack variables above its initial stack setting, then we know
199 // it's accessing variables passed into it, which means the data is on the
201 // When we're called, on the stack (high to low) we have the parameters, the
202 // return address, then the saved ebp. Save the pointer to the return address.
203 // EBC code knows that's there, so should look above it for function parameters.
204 // The offset is the size of locals (VMContext + Addr + saved ebp).
205 // Note that the interpreter assumes there is a 16 bytes of return address on
206 // the stack too, so adjust accordingly.
207 // VmContext.HighStackBottom = (UINTN)(Addr + sizeof (VmContext) + sizeof (Addr));
211 // Begin executing the EBC code
213 EbcExecute (&VmContext
);
216 // Return the value in R[7] unless there was an error
218 ReturnEBCStack(StackIndex
);
219 return (UINT64
) VmContext
.Gpr
[7];
224 Begin executing an EBC image. The address of the entry point is passed
225 in via a processor register, so we'll need to make a call to get the
228 @param ImageHandle image handle for the EBC application we're executing
229 @param SystemTable standard system table passed into an driver's entry
232 @return The value returned by the EBC application we're going to run.
236 ExecuteEbcImageEntryPoint (
237 IN EFI_HANDLE ImageHandle
,
238 IN EFI_SYSTEM_TABLE
*SystemTable
242 // Create a new VM context on the stack
244 VM_CONTEXT VmContext
;
250 // Get the EBC entry point from the processor register. Make sure you don't
251 // call any functions before this or you could mess up the register the
252 // entry point is passed in.
254 Addr
= EbcLLGetEbcEntryPoint ();
257 // Now clear out our context
259 ZeroMem ((VOID
*) &VmContext
, sizeof (VM_CONTEXT
));
262 // Save the image handle so we can track the thunks created for this image
264 VmContext
.ImageHandle
= ImageHandle
;
265 VmContext
.SystemTable
= SystemTable
;
268 // Set the VM instruction pointer to the correct location in memory.
270 VmContext
.Ip
= (VMIP
) Addr
;
273 // Initialize the stack pointer for the EBC. Get the current system stack
274 // pointer and adjust it down by the max needed for the interpreter.
276 Addr
= EbcLLGetStackPointer ();
278 Status
= GetEBCStack(ImageHandle
, &VmContext
.StackPool
, &StackIndex
);
279 if (EFI_ERROR(Status
)) {
282 VmContext
.StackTop
= (UINT8
*)VmContext
.StackPool
+ (STACK_REMAIN_SIZE
);
283 VmContext
.Gpr
[0] = (UINT64
) ((UINT8
*)VmContext
.StackPool
+ STACK_POOL_SIZE
);
284 VmContext
.HighStackBottom
= (UINTN
) VmContext
.Gpr
[0];
285 VmContext
.Gpr
[0] -= sizeof (UINTN
);
289 // Put a magic value in the stack gap, then adjust down again
291 *(UINTN
*) (UINTN
) (VmContext
.Gpr
[0]) = (UINTN
) VM_STACK_KEY_VALUE
;
292 VmContext
.StackMagicPtr
= (UINTN
*) (UINTN
) VmContext
.Gpr
[0];
295 // Align the stack on a natural boundary
296 VmContext
.Gpr
[0] &= ~(sizeof(UINTN
) - 1);
298 VmContext
.LowStackTop
= (UINTN
) VmContext
.Gpr
[0];
301 // Simply copy the image handle and system table onto the EBC stack.
302 // Greatly simplifies things by not having to spill the args.
304 PushU64 (&VmContext
, (UINT64
) SystemTable
);
305 PushU64 (&VmContext
, (UINT64
) ImageHandle
);
308 // VM pushes 16-bytes for return address. Simulate that here.
310 PushU64 (&VmContext
, (UINT64
) 0);
311 PushU64 (&VmContext
, (UINT64
) 0x1234567887654321ULL
);
314 // For x64, this is where we say our return address is
316 VmContext
.StackRetAddr
= (UINT64
) VmContext
.Gpr
[0];
319 // Entry function needn't access high stack context, simply
320 // put the stack pointer here.
324 // Begin executing the EBC code
326 EbcExecute (&VmContext
);
329 // Return the value in R[7] unless there was an error
331 ReturnEBCStack(StackIndex
);
332 return (UINT64
) VmContext
.Gpr
[7];
337 Create thunks for an EBC image entry point, or an EBC protocol service.
339 @param ImageHandle Image handle for the EBC image. If not null, then
340 we're creating a thunk for an image entry point.
341 @param EbcEntryPoint Address of the EBC code that the thunk is to call
342 @param Thunk Returned thunk we create here
343 @param Flags Flags indicating options for creating the thunk
345 @retval EFI_SUCCESS The thunk was created successfully.
346 @retval EFI_INVALID_PARAMETER The parameter of EbcEntryPoint is not 16-bit
348 @retval EFI_OUT_OF_RESOURCES There is not enough memory to created the EBC
350 @retval EFI_BUFFER_TOO_SMALL EBC_THUNK_SIZE is not larger enough.
355 IN EFI_HANDLE ImageHandle
,
356 IN VOID
*EbcEntryPoint
,
369 // Check alignment of pointer to EBC code
371 if ((UINT32
) (UINTN
) EbcEntryPoint
& 0x01) {
372 return EFI_INVALID_PARAMETER
;
375 Size
= EBC_THUNK_SIZE
;
378 Ptr
= AllocatePool (Size
);
381 return EFI_OUT_OF_RESOURCES
;
384 // Print(L"Allocate TH: 0x%X\n", (UINT32)Ptr);
386 // Save the start address so we can add a pointer to it to a list later.
391 // Give them the address of our buffer we're going to fix up
393 *Thunk
= (VOID
*) Ptr
;
396 // Add a magic code here to help the VM recognize the thunk..
397 // mov rax, ca112ebccall2ebch => 48 B8 BC 2E 11 CA BC 2E 11 CA
405 Addr
= (UINT64
) 0xCA112EBCCA112EBCULL
;
406 for (Index
= 0; Index
< sizeof (Addr
); Index
++) {
407 *Ptr
= (UINT8
) (UINTN
) Addr
;
414 // Add code bytes to load up a processor register with the EBC entry point.
415 // mov r10, 123456789abcdef0h => 49 BA F0 DE BC 9A 78 56 34 12
416 // The first 8 bytes of the thunk entry is the address of the EBC
425 Addr
= (UINT64
) EbcEntryPoint
;
426 for (Index
= 0; Index
< sizeof (Addr
); Index
++) {
427 *Ptr
= (UINT8
) (UINTN
) Addr
;
434 // Stick in a load of ecx with the address of appropriate VM function.
435 // Using r11 because it's a volatile register and won't be used in this
437 // mov r11 123456789abcdef0h => 49 BB F0 DE BC 9A 78 56 34 12
439 if ((Flags
& FLAG_THUNK_ENTRY_POINT
) != 0) {
440 Addr
= (UINTN
) ExecuteEbcImageEntryPoint
;
442 Addr
= (UINTN
) EbcInterpret
;
446 // mov r11 Addr => 0x49 0xBB
454 for (Index
= 0; Index
< sizeof (Addr
); Index
++) {
461 // Stick in jump opcode bytes for jmp r11 => 0x41 0xFF 0xE3
473 // Double check that our defined size is ok (application error)
477 return EFI_BUFFER_TOO_SMALL
;
480 // Add the thunk to the list for this image. Do this last since the add
481 // function flushes the cache for us.
483 EbcAddImageThunk (ImageHandle
, (VOID
*) ThunkBase
, ThunkSize
);
490 This function is called to execute an EBC CALLEX instruction.
491 The function check the callee's content to see whether it is common native
492 code or a thunk to another piece of EBC code.
493 If the callee is common native code, use EbcLLCAllEXASM to manipulate,
494 otherwise, set the VM->IP to target EBC code directly to avoid another VM
495 be startup which cost time and stack space.
497 @param VmPtr Pointer to a VM context.
498 @param FuncAddr Callee's address
499 @param NewStackPointer New stack pointer after the call
500 @param FramePtr New frame pointer after the call
501 @param Size The size of call instruction
506 IN VM_CONTEXT
*VmPtr
,
508 IN UINTN NewStackPointer
,
520 // Processor specific code to check whether the callee is a thunk to EBC.
522 if (*((UINT8
*)FuncAddr
) != 0x48) {
526 if (*((UINT8
*)FuncAddr
+ 1) != 0xB8) {
530 if (*((UINT8
*)FuncAddr
+ 2) != 0xBC) {
534 if (*((UINT8
*)FuncAddr
+ 3) != 0x2E) {
538 if (*((UINT8
*)FuncAddr
+ 4) != 0x11) {
542 if (*((UINT8
*)FuncAddr
+ 5) != 0xCA) {
546 if (*((UINT8
*)FuncAddr
+ 6) != 0xBC) {
550 if (*((UINT8
*)FuncAddr
+ 7) != 0x2E) {
554 if (*((UINT8
*)FuncAddr
+ 8) != 0x11) {
558 if (*((UINT8
*)FuncAddr
+ 9) != 0xCA) {
562 if (*((UINT8
*)FuncAddr
+ 10) != 0x49) {
566 if (*((UINT8
*)FuncAddr
+ 11) != 0xBA) {
571 CopyMem (&TargetEbcAddr
, (UINT8
*)FuncAddr
+ 12, 8);
576 // The callee is a thunk to EBC, adjust the stack pointer down 16 bytes and
577 // put our return address and frame pointer on the VM stack.
578 // Then set the VM's IP to new EBC code.
581 VmWriteMemN (VmPtr
, (UINTN
) VmPtr
->Gpr
[0], (UINTN
) FramePtr
);
582 VmPtr
->FramePtr
= (VOID
*) (UINTN
) VmPtr
->Gpr
[0];
584 VmWriteMem64 (VmPtr
, (UINTN
) VmPtr
->Gpr
[0], (UINT64
) (VmPtr
->Ip
+ Size
));
586 VmPtr
->Ip
= (VMIP
) (UINTN
) TargetEbcAddr
;
589 // The callee is not a thunk to EBC, call native code.
591 EbcLLCALLEXNative (FuncAddr
, NewStackPointer
, FramePtr
);
594 // Get return value and advance the IP.
596 VmPtr
->Gpr
[7] = EbcLLGetReturnValue ();