]> git.proxmox.com Git - mirror_edk2.git/blob - MdeModulePkg/Universal/EbcDxe/EbcExecute.h
bd405d3fac2d95a26c46bbefa0f8a79b97b77293
[mirror_edk2.git] / MdeModulePkg / Universal / EbcDxe / EbcExecute.h
1 /** @file
2 Header file for Virtual Machine support. Contains EBC defines that can
3 be of use to a disassembler for the most part. Also provides function
4 prototypes for VM functions.
5
6 Copyright (c) 2006 - 2010, Intel Corporation. All rights reserved.<BR>
7 This program and the accompanying materials
8 are licensed and made available under the terms and conditions of the BSD License
9 which accompanies this distribution. The full text of the license may be found at
10 http://opensource.org/licenses/bsd-license.php
11
12 THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
13 WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
14
15 **/
16
17 #ifndef _EBC_EXECUTE_H_
18 #define _EBC_EXECUTE_H_
19
20 //
21 // VM major/minor version
22 //
23 #define VM_MAJOR_VERSION 1
24 #define VM_MINOR_VERSION 0
25
26 //
27 // Macros to check and set alignment
28 //
29 #define ASSERT_ALIGNED(addr, size) ASSERT (!((UINT32) (addr) & (size - 1)))
30 #define IS_ALIGNED(addr, size) !((UINT32) (addr) & (size - 1))
31
32 //
33 // Define a macro to get the operand. Then we can change it to be either a
34 // direct read or have it call a function to read memory.
35 //
36 #define GETOPERANDS(pVM) (UINT8) (*(UINT8 *) (pVM->Ip + 1))
37 #define GETOPCODE(pVM) (UINT8) (*(UINT8 *) pVM->Ip)
38
39 //
40 // Bit masks for opcode encodings
41 //
42 #define OPCODE_M_OPCODE 0x3F // bits of interest for first level decode
43 #define OPCODE_M_IMMDATA 0x80
44 #define OPCODE_M_IMMDATA64 0x40
45 #define OPCODE_M_64BIT 0x40 // for CMP
46 #define OPCODE_M_RELADDR 0x10 // for CALL instruction
47 #define OPCODE_M_CMPI32_DATA 0x80 // for CMPI
48 #define OPCODE_M_CMPI64 0x40 // for CMPI 32 or 64 bit comparison
49 #define OPERAND_M_MOVIN_N 0x80
50 #define OPERAND_M_CMPI_INDEX 0x10
51
52 //
53 // Masks for instructions that encode presence of indexes for operand1 and/or
54 // operand2.
55 //
56 #define OPCODE_M_IMMED_OP1 0x80
57 #define OPCODE_M_IMMED_OP2 0x40
58
59 //
60 // Bit masks for operand encodings
61 //
62 #define OPERAND_M_INDIRECT1 0x08
63 #define OPERAND_M_INDIRECT2 0x80
64 #define OPERAND_M_OP1 0x07
65 #define OPERAND_M_OP2 0x70
66
67 //
68 // Masks for data manipulation instructions
69 //
70 #define DATAMANIP_M_64 0x40 // 64-bit width operation
71 #define DATAMANIP_M_IMMDATA 0x80
72
73 //
74 // For MOV instructions, need a mask for the opcode when immediate
75 // data applies to R2.
76 //
77 #define OPCODE_M_IMMED_OP2 0x40
78
79 //
80 // The MOVI/MOVIn instructions use bit 6 of operands byte to indicate
81 // if an index is present. Then bits 4 and 5 are used to indicate the width
82 // of the move.
83 //
84 #define MOVI_M_IMMDATA 0x40
85 #define MOVI_M_DATAWIDTH 0xC0
86 #define MOVI_DATAWIDTH16 0x40
87 #define MOVI_DATAWIDTH32 0x80
88 #define MOVI_DATAWIDTH64 0xC0
89 #define MOVI_M_MOVEWIDTH 0x30
90 #define MOVI_MOVEWIDTH8 0x00
91 #define MOVI_MOVEWIDTH16 0x10
92 #define MOVI_MOVEWIDTH32 0x20
93 #define MOVI_MOVEWIDTH64 0x30
94
95 //
96 // Masks for CALL instruction encodings
97 //
98 #define OPERAND_M_RELATIVE_ADDR 0x10
99 #define OPERAND_M_NATIVE_CALL 0x20
100
101 //
102 // Masks for decoding push/pop instructions
103 //
104 #define PUSHPOP_M_IMMDATA 0x80 // opcode bit indicating immediate data
105 #define PUSHPOP_M_64 0x40 // opcode bit indicating 64-bit operation
106 //
107 // Mask for operand of JMP instruction
108 //
109 #define JMP_M_RELATIVE 0x10
110 #define JMP_M_CONDITIONAL 0x80
111 #define JMP_M_CS 0x40
112
113 //
114 // Macros to determine if a given operand is indirect
115 //
116 #define OPERAND1_INDIRECT(op) ((op) & OPERAND_M_INDIRECT1)
117 #define OPERAND2_INDIRECT(op) ((op) & OPERAND_M_INDIRECT2)
118
119 //
120 // Macros to extract the operands from second byte of instructions
121 //
122 #define OPERAND1_REGNUM(op) ((op) & OPERAND_M_OP1)
123 #define OPERAND2_REGNUM(op) (((op) & OPERAND_M_OP2) >> 4)
124
125 #define OPERAND1_CHAR(op) ('0' + OPERAND1_REGNUM (op))
126 #define OPERAND2_CHAR(op) ('0' + OPERAND2_REGNUM (op))
127
128 #define OPERAND1_REGDATA(pvm, op) pvm->Gpr[OPERAND1_REGNUM (op)]
129 #define OPERAND2_REGDATA(pvm, op) pvm->Gpr[OPERAND2_REGNUM (op)]
130
131 //
132 // Condition masks usually for byte 1 encodings of code
133 //
134 #define CONDITION_M_CONDITIONAL 0x80
135 #define CONDITION_M_CS 0x40
136
137 //
138 // Bits in the VM->StopFlags field
139 //
140 #define STOPFLAG_APP_DONE 0x0001
141 #define STOPFLAG_BREAKPOINT 0x0002
142 #define STOPFLAG_INVALID_BREAK 0x0004
143 #define STOPFLAG_BREAK_ON_CALLEX 0x0008
144
145 //
146 // Masks for working with the VM flags register
147 //
148 #define VMFLAGS_CC 0x0001 // condition flag
149 #define VMFLAGS_STEP 0x0002 // step instruction mode
150 #define VMFLAGS_ALL_VALID (VMFLAGS_CC | VMFLAGS_STEP)
151
152 //
153 // Macros for operating on the VM flags register
154 //
155 #define VMFLAG_SET(pVM, Flag) (pVM->Flags |= (Flag))
156 #define VMFLAG_ISSET(pVM, Flag) ((pVM->Flags & (Flag)) ? 1 : 0)
157 #define VMFLAG_CLEAR(pVM, Flag) (pVM->Flags &= ~(Flag))
158
159 //
160 // Debug macro
161 //
162 #define EBCMSG(s) gST->ConOut->OutputString (gST->ConOut, s)
163
164 //
165 // Define OPCODES
166 //
167 #define OPCODE_BREAK 0x00
168 #define OPCODE_JMP 0x01
169 #define OPCODE_JMP8 0x02
170 #define OPCODE_CALL 0x03
171 #define OPCODE_RET 0x04
172 #define OPCODE_CMPEQ 0x05
173 #define OPCODE_CMPLTE 0x06
174 #define OPCODE_CMPGTE 0x07
175 #define OPCODE_CMPULTE 0x08
176 #define OPCODE_CMPUGTE 0x09
177 #define OPCODE_NOT 0x0A
178 #define OPCODE_NEG 0x0B
179 #define OPCODE_ADD 0x0C
180 #define OPCODE_SUB 0x0D
181 #define OPCODE_MUL 0x0E
182 #define OPCODE_MULU 0x0F
183 #define OPCODE_DIV 0x10
184 #define OPCODE_DIVU 0x11
185 #define OPCODE_MOD 0x12
186 #define OPCODE_MODU 0x13
187 #define OPCODE_AND 0x14
188 #define OPCODE_OR 0x15
189 #define OPCODE_XOR 0x16
190 #define OPCODE_SHL 0x17
191 #define OPCODE_SHR 0x18
192 #define OPCODE_ASHR 0x19
193 #define OPCODE_EXTNDB 0x1A
194 #define OPCODE_EXTNDW 0x1B
195 #define OPCODE_EXTNDD 0x1C
196 #define OPCODE_MOVBW 0x1D
197 #define OPCODE_MOVWW 0x1E
198 #define OPCODE_MOVDW 0x1F
199 #define OPCODE_MOVQW 0x20
200 #define OPCODE_MOVBD 0x21
201 #define OPCODE_MOVWD 0x22
202 #define OPCODE_MOVDD 0x23
203 #define OPCODE_MOVQD 0x24
204 #define OPCODE_MOVSNW 0x25 // Move signed natural with word index
205 #define OPCODE_MOVSND 0x26 // Move signed natural with dword index
206 //
207 // #define OPCODE_27 0x27
208 //
209 #define OPCODE_MOVQQ 0x28 // Does this go away?
210 #define OPCODE_LOADSP 0x29
211 #define OPCODE_STORESP 0x2A
212 #define OPCODE_PUSH 0x2B
213 #define OPCODE_POP 0x2C
214 #define OPCODE_CMPIEQ 0x2D
215 #define OPCODE_CMPILTE 0x2E
216 #define OPCODE_CMPIGTE 0x2F
217 #define OPCODE_CMPIULTE 0x30
218 #define OPCODE_CMPIUGTE 0x31
219 #define OPCODE_MOVNW 0x32
220 #define OPCODE_MOVND 0x33
221 //
222 // #define OPCODE_34 0x34
223 //
224 #define OPCODE_PUSHN 0x35
225 #define OPCODE_POPN 0x36
226 #define OPCODE_MOVI 0x37
227 #define OPCODE_MOVIN 0x38
228 #define OPCODE_MOVREL 0x39
229
230 /**
231 Execute an EBC image from an entry point or from a published protocol.
232
233 @param VmPtr A pointer to a VM context.
234
235 @retval EFI_UNSUPPORTED At least one of the opcodes is not supported.
236 @retval EFI_SUCCESS All of the instructions are executed successfully.
237
238 **/
239 EFI_STATUS
240 EbcExecute (
241 IN VM_CONTEXT *VmPtr
242 );
243
244
245
246 /**
247 Returns the version of the EBC virtual machine.
248
249 @return The 64-bit version of EBC virtual machine.
250
251 **/
252 UINT64
253 GetVmVersion (
254 VOID
255 );
256
257 /**
258 Writes UINTN data to memory address.
259
260 This routine is called by the EBC data
261 movement instructions that write to memory. Since these writes
262 may be to the stack, which looks like (high address on top) this,
263
264 [EBC entry point arguments]
265 [VM stack]
266 [EBC stack]
267
268 we need to detect all attempts to write to the EBC entry point argument
269 stack area and adjust the address (which will initially point into the
270 VM stack) to point into the EBC entry point arguments.
271
272 @param VmPtr A pointer to a VM context.
273 @param Addr Address to write to.
274 @param Data Value to write to Addr.
275
276 @retval EFI_SUCCESS The instruction is executed successfully.
277 @retval Other Some error occurs when writing data to the address.
278
279 **/
280 EFI_STATUS
281 VmWriteMemN (
282 IN VM_CONTEXT *VmPtr,
283 IN UINTN Addr,
284 IN UINTN Data
285 );
286
287 /**
288 Writes 64-bit data to memory address.
289
290 This routine is called by the EBC data
291 movement instructions that write to memory. Since these writes
292 may be to the stack, which looks like (high address on top) this,
293
294 [EBC entry point arguments]
295 [VM stack]
296 [EBC stack]
297
298 we need to detect all attempts to write to the EBC entry point argument
299 stack area and adjust the address (which will initially point into the
300 VM stack) to point into the EBC entry point arguments.
301
302 @param VmPtr A pointer to a VM context.
303 @param Addr Address to write to.
304 @param Data Value to write to Addr.
305
306 @retval EFI_SUCCESS The instruction is executed successfully.
307 @retval Other Some error occurs when writing data to the address.
308
309 **/
310 EFI_STATUS
311 VmWriteMem64 (
312 IN VM_CONTEXT *VmPtr,
313 IN UINTN Addr,
314 IN UINT64 Data
315 );
316
317 //
318 // Define a protocol for an EBC VM test interface.
319 //
320 #define EFI_EBC_VM_TEST_PROTOCOL_GUID \
321 { \
322 0xAAEACCFDL, 0xF27B, 0x4C17, { 0xB6, 0x10, 0x75, 0xCA, 0x1F, 0x2D, 0xFB, 0x52 } \
323 }
324
325 //
326 // Define for forward reference.
327 //
328 typedef struct _EFI_EBC_VM_TEST_PROTOCOL EFI_EBC_VM_TEST_PROTOCOL;
329
330 typedef
331 EFI_STATUS
332 (*EBC_VM_TEST_EXECUTE) (
333 IN EFI_EBC_VM_TEST_PROTOCOL * This,
334 IN VM_CONTEXT * VmPtr,
335 IN OUT UINTN *InstructionCount
336 );
337
338 typedef
339 EFI_STATUS
340 (*EBC_VM_TEST_ASM) (
341 IN EFI_EBC_VM_TEST_PROTOCOL * This,
342 IN CHAR16 *AsmText,
343 IN OUT INT8 *Buffer,
344 IN OUT UINTN *BufferLen
345 );
346
347 typedef
348 EFI_STATUS
349 (*EBC_VM_TEST_DASM) (
350 IN EFI_EBC_VM_TEST_PROTOCOL * This,
351 IN OUT CHAR16 *AsmText,
352 IN OUT INT8 *Buffer,
353 IN OUT UINTN *Len
354 );
355
356 //
357 // Prototype for the actual EBC test protocol interface
358 //
359 struct _EFI_EBC_VM_TEST_PROTOCOL {
360 EBC_VM_TEST_EXECUTE Execute;
361 EBC_VM_TEST_ASM Assemble;
362 EBC_VM_TEST_DASM Disassemble;
363 };
364
365 /**
366 Given a pointer to a new VM context, execute one or more instructions. This
367 function is only used for test purposes via the EBC VM test protocol.
368
369 @param This A pointer to the EFI_EBC_VM_TEST_PROTOCOL structure.
370 @param VmPtr A pointer to a VM context.
371 @param InstructionCount A pointer to a UINTN value holding the number of
372 instructions to execute. If it holds value of 0,
373 then the instruction to be executed is 1.
374
375 @retval EFI_UNSUPPORTED At least one of the opcodes is not supported.
376 @retval EFI_SUCCESS All of the instructions are executed successfully.
377
378 **/
379 EFI_STATUS
380 EbcExecuteInstructions (
381 IN EFI_EBC_VM_TEST_PROTOCOL *This,
382 IN VM_CONTEXT *VmPtr,
383 IN OUT UINTN *InstructionCount
384 );
385
386 #endif // ifndef _EBC_EXECUTE_H_