VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/bootsectors/bs3-cpu-generated-1-template.c@ 66196

Last change on this file since 66196 was 66188, checked in by vboxsync, 8 years ago

build fix

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 83.7 KB
Line 
1/* $Id: bs3-cpu-generated-1-template.c 66188 2017-03-21 19:34:39Z vboxsync $ */
2/** @file
3 * BS3Kit - bs3-cpu-generated-1, C code template.
4 */
5
6/*
7 * Copyright (C) 2007-2017 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27#ifndef BS3_INSTANTIATING_CMN
28# error "BS3_INSTANTIATING_CMN not defined"
29#endif
30
31
32/*********************************************************************************************************************************
33* Header Files *
34*********************************************************************************************************************************/
35#include <iprt/asm.h>
36#include <iprt/asm-amd64-x86.h>
37
38#include "bs3-cpu-generated-1.h"
39
40
41/*********************************************************************************************************************************
42* Defined Constants And Macros *
43*********************************************************************************************************************************/
44#define P_CS X86_OP_PRF_CS
45#define P_SS X86_OP_PRF_SS
46#define P_DS X86_OP_PRF_DS
47#define P_ES X86_OP_PRF_ES
48#define P_FS X86_OP_PRF_FS
49#define P_GS X86_OP_PRF_GS
50#define P_OZ X86_OP_PRF_SIZE_OP
51#define P_AZ X86_OP_PRF_SIZE_ADDR
52#define P_LK X86_OP_PRF_LOCK
53#define P_RN X86_OP_PRF_REPNZ
54#define P_RZ X86_OP_PRF_REPZ
55
56#define REX_WRBX (X86_OP_REX_W | X86_OP_REX_R | X86_OP_REX_B | X86_OP_REX_X)
57#define REX_W___ (X86_OP_REX_W)
58#define REX_WR__ (X86_OP_REX_W | X86_OP_REX_R)
59#define REX_W_B_ (X86_OP_REX_W | X86_OP_REX_B)
60#define REX_W__X (X86_OP_REX_W | X86_OP_REX_X)
61#define REX_WRB_ (X86_OP_REX_W | X86_OP_REX_R | X86_OP_REX_B)
62#define REX_WR_X (X86_OP_REX_W | X86_OP_REX_R | X86_OP_REX_X)
63#define REX_W_BX (X86_OP_REX_W | X86_OP_REX_B | X86_OP_REX_X)
64#define REX__R__ (X86_OP_REX_R)
65#define REX__RB_ (X86_OP_REX_R | X86_OP_REX_B)
66#define REX__R_X (X86_OP_REX_R | X86_OP_REX_X)
67#define REX__RBX (X86_OP_REX_R | X86_OP_REX_B | X86_OP_REX_X)
68#define REX___B_ (X86_OP_REX_B)
69#define REX___BX (X86_OP_REX_B | X86_OP_REX_X)
70#define REX____X (X86_OP_REX_X)
71#define REX_____ (0x40)
72
73
74/** @def BS3CG1_DPRINTF
75 * Debug print macro.
76 */
77#if 0
78# define BS3CG1_DPRINTF(a_ArgList) Bs3TestPrintf a_ArgList
79# define BS3CG1_DEBUG_CTX_MOD
80#else
81# define BS3CG1_DPRINTF(a_ArgList) do { } while (0)
82#endif
83
84
85
86/*********************************************************************************************************************************
87* Structures and Typedefs *
88*********************************************************************************************************************************/
89/** Operand value location. */
90typedef enum BS3CG1OPLOC
91{
92 BS3CG1OPLOC_INVALID = 0,
93 BS3CG1OPLOC_CTX,
94 BS3CG1OPLOC_IMM,
95 BS3CG1OPLOC_MEM,
96 BS3CG1OPLOC_MEM_RW,
97 BS3CG1OPLOC_END
98} BS3CG1OPLOC;
99
100/**
101 * The state.
102 */
103typedef struct BS3CG1STATE
104{
105 /** @name Instruction details (expanded from BS3CG1INSTR).
106 * @{ */
107 /** Pointer to the mnemonic string (not terminated) (g_achBs3Cg1Mnemonics). */
108 const char BS3_FAR *pchMnemonic;
109 /** Pointer to the test header. */
110 PCBS3CG1TESTHDR pTestHdr;
111 /** Pointer to the per operand flags (g_abBs3Cg1Operands). */
112 const uint8_t BS3_FAR *pabOperands;
113 /** Opcode bytes (g_abBs3Cg1Opcodes). */
114 const uint8_t BS3_FAR *pabOpcodes;
115 /** The current instruction number in the input array (for error reporting). */
116 uint32_t iInstr;
117
118 /** The instruction flags. */
119 uint32_t fFlags;
120 /** The encoding. */
121 BS3CG1ENC enmEncoding;
122 /** The CPU test / CPU ID. */
123 BS3CG1CPU enmCpuTest;
124 /** Per operand flags. */
125 BS3CG1OP aenmOperands[4];
126 /** Opcode bytes. */
127 uint8_t abOpcodes[4];
128
129 /** The length of the mnemonic. */
130 uint8_t cchMnemonic;
131 /** Whether to advance the mnemonic pointer or not. */
132 uint8_t fAdvanceMnemonic;
133 /** The number of opcode bytes. */
134 uint8_t cbOpcodes;
135 /** Number of operands. */
136 uint8_t cOperands;
137 /** @} */
138
139 /** Operand size in bytes (0 if not applicable). */
140 uint8_t cbOperand;
141 /** Target ring (0..3). */
142 uint8_t uCpl;
143
144 /** The current test number. */
145 uint8_t iTest;
146
147 /** Target mode (g_bBs3CurrentMode). */
148 uint8_t bMode;
149
150
151 /** @name Current encoded instruction.
152 * @{ */
153 /** The size of the current instruction that we're testing. */
154 uint8_t cbCurInstr;
155 /** The size the prefixes. */
156 uint8_t cbCurPrefix;
157 /** The offset into abCurInstr of the immediate. */
158 uint8_t offCurImm;
159 /** Buffer for assembling the current instruction. */
160 uint8_t abCurInstr[27];
161
162 /** Set if the encoding can't be tested in the same ring as this test code.
163 * This is used to deal with encodings modifying SP/ESP/RSP. */
164 bool fSameRingNotOkay;
165 /** The aOperands index of the modrm.reg operand (if applicable). */
166 uint8_t iRegOp;
167 /** The aOperands index of the modrm.rm operand (if applicable). */
168 uint8_t iRmOp;
169
170 /** Operands details. */
171 struct
172 {
173 uint8_t cbOp;
174 /** BS3CG1OPLOC_XXX. */
175 uint8_t enmLocation;
176 /** The BS3CG1DST value for this field.
177 * Set to BS3CG1DST_INVALID if memory or immediate. */
178 uint8_t idxField;
179 /** Depends on enmLocation.
180 * - BS3CG1OPLOC_IMM: offset relative to start of the instruction.
181 * - BS3CG1OPLOC_MEM: offset should be subtracted from &pbDataPg[_4K].
182 * - BS3CG1OPLOC_MEM_RW: offset should be subtracted from &pbDataPg[_4K].
183 * - BS3CG1OPLOC_CTX: not used (use idxField instead).
184 */
185 uint8_t off;
186 } aOperands[4];
187 /** @} */
188
189 /** Page to put code in. When paging is enabled, the page before and after
190 * are marked not-present. */
191 uint8_t BS3_FAR *pbCodePg;
192 /** The flat address corresponding to pbCodePg. */
193 uintptr_t uCodePgFlat;
194 /** The 16-bit address corresponding to pbCodePg if relevant for bMode. */
195 RTFAR16 CodePgFar;
196 /** The IP/EIP/RIP value for pbCodePg[0] relative to CS (bMode). */
197 uintptr_t CodePgRip;
198
199 /** Page for placing data operands in. When paging is enabled, the page before
200 * and after are marked not-present. */
201 uint8_t BS3_FAR *pbDataPg;
202 /** The flat address corresponding to pbDataPg. */
203 uintptr_t uDataPgFlat;
204 /** The 16-bit address corresponding to pbDataPg. */
205 RTFAR16 DataPgFar;
206
207 /** The name corresponding to bMode. */
208 const char BS3_FAR *pszMode;
209 /** The short name corresponding to bMode. */
210 const char BS3_FAR *pszModeShort;
211
212 /** @name Expected result (modifiable by output program).
213 * @{ */
214 /** The expected exception based on operand values or result.
215 * UINT8_MAX if no special exception expected. */
216 uint8_t bValueXcpt;
217 /** @} */
218
219 /** The context we're working on. */
220 BS3REGCTX Ctx;
221 /** The trap context and frame. */
222 BS3TRAPFRAME TrapFrame;
223 /** Initial contexts, one for each ring. */
224 BS3REGCTX aInitialCtxs[4];
225
226 /** Memory operand scratch space. */
227 union
228 {
229 uint8_t ab[128];
230 uint16_t au16[128 / sizeof(uint16_t)];
231 uint32_t au32[128 / sizeof(uint32_t)];
232 uint64_t au64[128 / sizeof(uint64_t)];
233 } MemOp;
234
235 /** Array parallel to aInitialCtxs for saving segment registers. */
236 struct
237 {
238 RTSEL ds;
239 } aSavedSegRegs[4];
240
241} BS3CG1STATE;
242/** Pointer to the generated test state. */
243typedef BS3CG1STATE *PBS3CG1STATE;
244
245
246#define BS3CG1_PF_OZ UINT16_C(0x0001)
247#define BS3CG1_PF_AZ UINT16_C(0x0002)
248#define BS3CG1_PF_CS UINT16_C(0x0004)
249#define BS3CG1_PF_DS UINT16_C(0x0008)
250#define BS3CG1_PF_ES UINT16_C(0x0010)
251#define BS3CG1_PF_FS UINT16_C(0x0020)
252#define BS3CG1_PF_GS UINT16_C(0x0040)
253#define BS3CG1_PF_SS UINT16_C(0x0080)
254#define BS3CG1_PF_SEGS (BS3CG1_PF_CS | BS3CG1_PF_DS | BS3CG1_PF_ES | BS3CG1_PF_FS | BS3CG1_PF_GS | BS3CG1_PF_SS)
255#define BS3CG1_PF_MEM (BS3CG1_PF_SEGS | BS3CG1_PF_AZ)
256#define BS3CG1_PF_LK UINT16_C(0x0100)
257#define BS3CG1_PF_RN UINT16_C(0x0200)
258#define BS3CG1_PF_RZ UINT16_C(0x0400)
259#define BS3CG1_PF_W UINT16_C(0x0800) /**< REX.W */
260#define BS3CG1_PF_R UINT16_C(0x1000) /**< REX.R */
261#define BS3CG1_PF_B UINT16_C(0x2000) /**< REX.B */
262#define BS3CG1_PF_X UINT16_C(0x4000) /**< REX.X */
263
264
265/** Used in g_cbBs3Cg1DstFields to indicate that it's one of the 4 operands. */
266#define BS3CG1DSTSIZE_OPERAND UINT8_C(255)
267/** Used in g_cbBs3Cg1DstFields to indicate that the operand size determins
268 * the field size (2, 4, or 8). */
269#define BS3CG1DSTSIZE_OPERAND_SIZE_GRP UINT8_C(254)
270
271
272
273/*********************************************************************************************************************************
274* Global Variables *
275*********************************************************************************************************************************/
276/** Destination field sizes indexed by bBS3CG1DST.
277 * Zero means operand size sized. */
278static const uint8_t g_acbBs3Cg1DstFields[] =
279{
280 /* [BS3CG1DST_INVALID] = */ BS3CG1DSTSIZE_OPERAND,
281
282 /* [BS3CG1DST_OP1] = */ BS3CG1DSTSIZE_OPERAND,
283 /* [BS3CG1DST_OP2] = */ BS3CG1DSTSIZE_OPERAND,
284 /* [BS3CG1DST_OP3] = */ BS3CG1DSTSIZE_OPERAND,
285 /* [BS3CG1DST_OP4] = */ BS3CG1DSTSIZE_OPERAND,
286 /* [BS3CG1DST_EFL] = */ 4,
287 /* [BS3CG1DST_EFL_UNDEF]=*/ 4,
288
289 /* [BS3CG1DST_AL] = */ 1,
290 /* [BS3CG1DST_CL] = */ 1,
291 /* [BS3CG1DST_DL] = */ 1,
292 /* [BS3CG1DST_BL] = */ 1,
293 /* [BS3CG1DST_AH] = */ 1,
294 /* [BS3CG1DST_CH] = */ 1,
295 /* [BS3CG1DST_DH] = */ 1,
296 /* [BS3CG1DST_BH] = */ 1,
297 /* [BS3CG1DST_SPL] = */ 1,
298 /* [BS3CG1DST_BPL] = */ 1,
299 /* [BS3CG1DST_SIL] = */ 1,
300 /* [BS3CG1DST_DIL] = */ 1,
301 /* [BS3CG1DST_R8L] = */ 1,
302 /* [BS3CG1DST_R9L] = */ 1,
303 /* [BS3CG1DST_R10L] = */ 1,
304 /* [BS3CG1DST_R11L] = */ 1,
305 /* [BS3CG1DST_R12L] = */ 1,
306 /* [BS3CG1DST_R13L] = */ 1,
307 /* [BS3CG1DST_R14L] = */ 1,
308 /* [BS3CG1DST_R15L] = */ 1,
309
310 /* [BS3CG1DST_AX] = */ 2,
311 /* [BS3CG1DST_CX] = */ 2,
312 /* [BS3CG1DST_DX] = */ 2,
313 /* [BS3CG1DST_BX] = */ 2,
314 /* [BS3CG1DST_SP] = */ 2,
315 /* [BS3CG1DST_BP] = */ 2,
316 /* [BS3CG1DST_SI] = */ 2,
317 /* [BS3CG1DST_DI] = */ 2,
318 /* [BS3CG1DST_R8W] = */ 2,
319 /* [BS3CG1DST_R9W] = */ 2,
320 /* [BS3CG1DST_R10W] = */ 2,
321 /* [BS3CG1DST_R11W] = */ 2,
322 /* [BS3CG1DST_R12W] = */ 2,
323 /* [BS3CG1DST_R13W] = */ 2,
324 /* [BS3CG1DST_R14W] = */ 2,
325 /* [BS3CG1DST_R15W] = */ 2,
326
327 /* [BS3CG1DST_EAX] = */ 4,
328 /* [BS3CG1DST_ECX] = */ 4,
329 /* [BS3CG1DST_EDX] = */ 4,
330 /* [BS3CG1DST_EBX] = */ 4,
331 /* [BS3CG1DST_ESP] = */ 4,
332 /* [BS3CG1DST_EBP] = */ 4,
333 /* [BS3CG1DST_ESI] = */ 4,
334 /* [BS3CG1DST_EDI] = */ 4,
335 /* [BS3CG1DST_R8D] = */ 4,
336 /* [BS3CG1DST_R9D] = */ 4,
337 /* [BS3CG1DST_R10D] = */ 4,
338 /* [BS3CG1DST_R11D] = */ 4,
339 /* [BS3CG1DST_R12D] = */ 4,
340 /* [BS3CG1DST_R13D] = */ 4,
341 /* [BS3CG1DST_R14D] = */ 4,
342 /* [BS3CG1DST_R15D] = */ 4,
343
344 /* [BS3CG1DST_RAX] = */ 8,
345 /* [BS3CG1DST_RCX] = */ 8,
346 /* [BS3CG1DST_RDX] = */ 8,
347 /* [BS3CG1DST_RBX] = */ 8,
348 /* [BS3CG1DST_RSP] = */ 8,
349 /* [BS3CG1DST_RBP] = */ 8,
350 /* [BS3CG1DST_RSI] = */ 8,
351 /* [BS3CG1DST_RDI] = */ 8,
352 /* [BS3CG1DST_R8] = */ 8,
353 /* [BS3CG1DST_R9] = */ 8,
354 /* [BS3CG1DST_R10] = */ 8,
355 /* [BS3CG1DST_R11] = */ 8,
356 /* [BS3CG1DST_R12] = */ 8,
357 /* [BS3CG1DST_R13] = */ 8,
358 /* [BS3CG1DST_R14] = */ 8,
359 /* [BS3CG1DST_R15] = */ 8,
360
361 /* [BS3CG1DST_OZ_RAX] = */ BS3CG1DSTSIZE_OPERAND_SIZE_GRP,
362 /* [BS3CG1DST_OZ_RCX] = */ BS3CG1DSTSIZE_OPERAND_SIZE_GRP,
363 /* [BS3CG1DST_OZ_RDX] = */ BS3CG1DSTSIZE_OPERAND_SIZE_GRP,
364 /* [BS3CG1DST_OZ_RBX] = */ BS3CG1DSTSIZE_OPERAND_SIZE_GRP,
365 /* [BS3CG1DST_OZ_RSP] = */ BS3CG1DSTSIZE_OPERAND_SIZE_GRP,
366 /* [BS3CG1DST_OZ_RBP] = */ BS3CG1DSTSIZE_OPERAND_SIZE_GRP,
367 /* [BS3CG1DST_OZ_RSI] = */ BS3CG1DSTSIZE_OPERAND_SIZE_GRP,
368 /* [BS3CG1DST_OZ_RDI] = */ BS3CG1DSTSIZE_OPERAND_SIZE_GRP,
369 /* [BS3CG1DST_OZ_R8] = */ BS3CG1DSTSIZE_OPERAND_SIZE_GRP,
370 /* [BS3CG1DST_OZ_R9] = */ BS3CG1DSTSIZE_OPERAND_SIZE_GRP,
371 /* [BS3CG1DST_OZ_R10] = */ BS3CG1DSTSIZE_OPERAND_SIZE_GRP,
372 /* [BS3CG1DST_OZ_R11] = */ BS3CG1DSTSIZE_OPERAND_SIZE_GRP,
373 /* [BS3CG1DST_OZ_R12] = */ BS3CG1DSTSIZE_OPERAND_SIZE_GRP,
374 /* [BS3CG1DST_OZ_R13] = */ BS3CG1DSTSIZE_OPERAND_SIZE_GRP,
375 /* [BS3CG1DST_OZ_R14] = */ BS3CG1DSTSIZE_OPERAND_SIZE_GRP,
376 /* [BS3CG1DST_OZ_R15] = */ BS3CG1DSTSIZE_OPERAND_SIZE_GRP,
377
378 /* [BS3CG1DST_VALUE_XCPT] = */ 1,
379
380};
381
382/** Destination field offset indexed by bBS3CG1DST.
383 * Zero means operand size sized. */
384static const unsigned g_aoffBs3Cg1DstFields[] =
385{
386 /* [BS3CG1DST_INVALID] = */ ~0U,
387 /* [BS3CG1DST_OP1] = */ ~0U,
388 /* [BS3CG1DST_OP2] = */ ~0U,
389 /* [BS3CG1DST_OP3] = */ ~0U,
390 /* [BS3CG1DST_OP4] = */ ~0U,
391 /* [BS3CG1DST_EFL] = */ RT_OFFSETOF(BS3REGCTX, rflags),
392 /* [BS3CG1DST_EFL_UNDEF]=*/ ~0, /* special field */
393
394 /* [BS3CG1DST_AL] = */ RT_OFFSETOF(BS3REGCTX, rax.u8),
395 /* [BS3CG1DST_CL] = */ RT_OFFSETOF(BS3REGCTX, rcx.u8),
396 /* [BS3CG1DST_DL] = */ RT_OFFSETOF(BS3REGCTX, rdx.u8),
397 /* [BS3CG1DST_BL] = */ RT_OFFSETOF(BS3REGCTX, rbx.u8),
398 /* [BS3CG1DST_AH] = */ RT_OFFSETOF(BS3REGCTX, rax.b.bHi),
399 /* [BS3CG1DST_CH] = */ RT_OFFSETOF(BS3REGCTX, rcx.b.bHi),
400 /* [BS3CG1DST_DH] = */ RT_OFFSETOF(BS3REGCTX, rdx.b.bHi),
401 /* [BS3CG1DST_BH] = */ RT_OFFSETOF(BS3REGCTX, rbx.b.bHi),
402 /* [BS3CG1DST_SPL] = */ RT_OFFSETOF(BS3REGCTX, rsp.u8),
403 /* [BS3CG1DST_BPL] = */ RT_OFFSETOF(BS3REGCTX, rbp.u8),
404 /* [BS3CG1DST_SIL] = */ RT_OFFSETOF(BS3REGCTX, rsi.u8),
405 /* [BS3CG1DST_DIL] = */ RT_OFFSETOF(BS3REGCTX, rdi.u8),
406 /* [BS3CG1DST_R8L] = */ RT_OFFSETOF(BS3REGCTX, r8.u8),
407 /* [BS3CG1DST_R9L] = */ RT_OFFSETOF(BS3REGCTX, r9.u8),
408 /* [BS3CG1DST_R10L] = */ RT_OFFSETOF(BS3REGCTX, r10.u8),
409 /* [BS3CG1DST_R11L] = */ RT_OFFSETOF(BS3REGCTX, r11.u8),
410 /* [BS3CG1DST_R12L] = */ RT_OFFSETOF(BS3REGCTX, r12.u8),
411 /* [BS3CG1DST_R13L] = */ RT_OFFSETOF(BS3REGCTX, r13.u8),
412 /* [BS3CG1DST_R14L] = */ RT_OFFSETOF(BS3REGCTX, r14.u8),
413 /* [BS3CG1DST_R15L] = */ RT_OFFSETOF(BS3REGCTX, r15.u8),
414
415 /* [BS3CG1DST_AX] = */ RT_OFFSETOF(BS3REGCTX, rax.u16),
416 /* [BS3CG1DST_CX] = */ RT_OFFSETOF(BS3REGCTX, rcx.u16),
417 /* [BS3CG1DST_DX] = */ RT_OFFSETOF(BS3REGCTX, rdx.u16),
418 /* [BS3CG1DST_BX] = */ RT_OFFSETOF(BS3REGCTX, rbx.u16),
419 /* [BS3CG1DST_SP] = */ RT_OFFSETOF(BS3REGCTX, rsp.u16),
420 /* [BS3CG1DST_BP] = */ RT_OFFSETOF(BS3REGCTX, rbp.u16),
421 /* [BS3CG1DST_SI] = */ RT_OFFSETOF(BS3REGCTX, rsi.u16),
422 /* [BS3CG1DST_DI] = */ RT_OFFSETOF(BS3REGCTX, rdi.u16),
423 /* [BS3CG1DST_R8W] = */ RT_OFFSETOF(BS3REGCTX, r8.u16),
424 /* [BS3CG1DST_R9W] = */ RT_OFFSETOF(BS3REGCTX, r9.u16),
425 /* [BS3CG1DST_R10W] = */ RT_OFFSETOF(BS3REGCTX, r10.u16),
426 /* [BS3CG1DST_R11W] = */ RT_OFFSETOF(BS3REGCTX, r11.u16),
427 /* [BS3CG1DST_R12W] = */ RT_OFFSETOF(BS3REGCTX, r12.u16),
428 /* [BS3CG1DST_R13W] = */ RT_OFFSETOF(BS3REGCTX, r13.u16),
429 /* [BS3CG1DST_R14W] = */ RT_OFFSETOF(BS3REGCTX, r14.u16),
430 /* [BS3CG1DST_R15W] = */ RT_OFFSETOF(BS3REGCTX, r15.u16),
431
432 /* [BS3CG1DST_EAX] = */ RT_OFFSETOF(BS3REGCTX, rax.u32),
433 /* [BS3CG1DST_ECX] = */ RT_OFFSETOF(BS3REGCTX, rcx.u32),
434 /* [BS3CG1DST_EDX] = */ RT_OFFSETOF(BS3REGCTX, rdx.u32),
435 /* [BS3CG1DST_EBX] = */ RT_OFFSETOF(BS3REGCTX, rbx.u32),
436 /* [BS3CG1DST_ESP] = */ RT_OFFSETOF(BS3REGCTX, rsp.u32),
437 /* [BS3CG1DST_EBP] = */ RT_OFFSETOF(BS3REGCTX, rbp.u32),
438 /* [BS3CG1DST_ESI] = */ RT_OFFSETOF(BS3REGCTX, rsi.u32),
439 /* [BS3CG1DST_EDI] = */ RT_OFFSETOF(BS3REGCTX, rdi.u32),
440 /* [BS3CG1DST_R8D] = */ RT_OFFSETOF(BS3REGCTX, r8.u32),
441 /* [BS3CG1DST_R9D] = */ RT_OFFSETOF(BS3REGCTX, r9.u32),
442 /* [BS3CG1DST_R10D] = */ RT_OFFSETOF(BS3REGCTX, r10.u32),
443 /* [BS3CG1DST_R11D] = */ RT_OFFSETOF(BS3REGCTX, r11.u32),
444 /* [BS3CG1DST_R12D] = */ RT_OFFSETOF(BS3REGCTX, r12.u32),
445 /* [BS3CG1DST_R13D] = */ RT_OFFSETOF(BS3REGCTX, r13.u32),
446 /* [BS3CG1DST_R14D] = */ RT_OFFSETOF(BS3REGCTX, r14.u32),
447 /* [BS3CG1DST_R15D] = */ RT_OFFSETOF(BS3REGCTX, r15.u32),
448
449 /* [BS3CG1DST_RAX] = */ RT_OFFSETOF(BS3REGCTX, rax.u64),
450 /* [BS3CG1DST_RCX] = */ RT_OFFSETOF(BS3REGCTX, rcx.u64),
451 /* [BS3CG1DST_RDX] = */ RT_OFFSETOF(BS3REGCTX, rdx.u64),
452 /* [BS3CG1DST_RBX] = */ RT_OFFSETOF(BS3REGCTX, rbx.u64),
453 /* [BS3CG1DST_RSP] = */ RT_OFFSETOF(BS3REGCTX, rsp.u64),
454 /* [BS3CG1DST_RBP] = */ RT_OFFSETOF(BS3REGCTX, rbp.u64),
455 /* [BS3CG1DST_RSI] = */ RT_OFFSETOF(BS3REGCTX, rsi.u64),
456 /* [BS3CG1DST_RDI] = */ RT_OFFSETOF(BS3REGCTX, rdi.u64),
457 /* [BS3CG1DST_R8] = */ RT_OFFSETOF(BS3REGCTX, r8.u64),
458 /* [BS3CG1DST_R9] = */ RT_OFFSETOF(BS3REGCTX, r9.u64),
459 /* [BS3CG1DST_R10] = */ RT_OFFSETOF(BS3REGCTX, r10.u64),
460 /* [BS3CG1DST_R11] = */ RT_OFFSETOF(BS3REGCTX, r11.u64),
461 /* [BS3CG1DST_R12] = */ RT_OFFSETOF(BS3REGCTX, r12.u64),
462 /* [BS3CG1DST_R13] = */ RT_OFFSETOF(BS3REGCTX, r13.u64),
463 /* [BS3CG1DST_R14] = */ RT_OFFSETOF(BS3REGCTX, r14.u64),
464 /* [BS3CG1DST_R15] = */ RT_OFFSETOF(BS3REGCTX, r15.u64),
465
466 /* [BS3CG1DST_OZ_RAX] = */ RT_OFFSETOF(BS3REGCTX, rax),
467 /* [BS3CG1DST_OZ_RCX] = */ RT_OFFSETOF(BS3REGCTX, rcx),
468 /* [BS3CG1DST_OZ_RDX] = */ RT_OFFSETOF(BS3REGCTX, rdx),
469 /* [BS3CG1DST_OZ_RBX] = */ RT_OFFSETOF(BS3REGCTX, rbx),
470 /* [BS3CG1DST_OZ_RSP] = */ RT_OFFSETOF(BS3REGCTX, rsp),
471 /* [BS3CG1DST_OZ_RBP] = */ RT_OFFSETOF(BS3REGCTX, rbp),
472 /* [BS3CG1DST_OZ_RSI] = */ RT_OFFSETOF(BS3REGCTX, rsi),
473 /* [BS3CG1DST_OZ_RDI] = */ RT_OFFSETOF(BS3REGCTX, rdi),
474 /* [BS3CG1DST_OZ_R8] = */ RT_OFFSETOF(BS3REGCTX, r8),
475 /* [BS3CG1DST_OZ_R9] = */ RT_OFFSETOF(BS3REGCTX, r9),
476 /* [BS3CG1DST_OZ_R10] = */ RT_OFFSETOF(BS3REGCTX, r10),
477 /* [BS3CG1DST_OZ_R11] = */ RT_OFFSETOF(BS3REGCTX, r11),
478 /* [BS3CG1DST_OZ_R12] = */ RT_OFFSETOF(BS3REGCTX, r12),
479 /* [BS3CG1DST_OZ_R13] = */ RT_OFFSETOF(BS3REGCTX, r13),
480 /* [BS3CG1DST_OZ_R14] = */ RT_OFFSETOF(BS3REGCTX, r14),
481 /* [BS3CG1DST_OZ_R15] = */ RT_OFFSETOF(BS3REGCTX, r15),
482
483 /* [BS3CG1DST_VALUE_XCPT] = */ ~0U,
484};
485
486#ifdef BS3CG1_DEBUG_CTX_MOD
487/** Destination field names. */
488static const struct { char sz[8]; } g_aszBs3Cg1DstFields[] =
489{
490 { "INVALID" },
491 { "OP1" },
492 { "OP2" },
493 { "OP3" },
494 { "OP4" },
495 { "EFL" },
496 { "EFL_UND" },
497
498 { "AL" },
499 { "CL" },
500 { "DL" },
501 { "BL" },
502 { "AH" },
503 { "CH" },
504 { "DH" },
505 { "BH" },
506 { "SPL" },
507 { "BPL" },
508 { "SIL" },
509 { "DIL" },
510 { "R8L" },
511 { "R9L" },
512 { "R10L" },
513 { "R11L" },
514 { "R12L" },
515 { "R13L" },
516 { "R14L" },
517 { "R15L" },
518
519 { "AX" },
520 { "CX" },
521 { "DX" },
522 { "BX" },
523 { "SP" },
524 { "BP" },
525 { "SI" },
526 { "DI" },
527 { "R8W" },
528 { "R9W" },
529 { "R10W" },
530 { "R11W" },
531 { "R12W" },
532 { "R13W" },
533 { "R14W" },
534 { "R15W" },
535
536 { "EAX" },
537 { "ECX" },
538 { "EDX" },
539 { "EBX" },
540 { "ESP" },
541 { "EBP" },
542 { "ESI" },
543 { "EDI" },
544 { "R8D" },
545 { "R9D" },
546 { "R10D" },
547 { "R11D" },
548 { "R12D" },
549 { "R13D" },
550 { "R14D" },
551 { "R15D" },
552
553 { "RAX" },
554 { "RCX" },
555 { "RDX" },
556 { "RBX" },
557 { "RSP" },
558 { "RBP" },
559 { "RSI" },
560 { "RDI" },
561 { "R8" },
562 { "R9" },
563 { "R10" },
564 { "R11" },
565 { "R12" },
566 { "R13" },
567 { "R14" },
568 { "R15" },
569
570 { "OZ_RAX" },
571 { "OZ_RCX" },
572 { "OZ_RDX" },
573 { "OZ_RBX" },
574 { "OZ_RSP" },
575 { "OZ_RBP" },
576 { "OZ_RSI" },
577 { "OZ_RDI" },
578 { "OZ_R8" },
579 { "OZ_R9" },
580 { "OZ_R10" },
581 { "OZ_R11" },
582 { "OZ_R12" },
583 { "OZ_R13" },
584 { "OZ_R14" },
585 { "OZ_R15" },
586
587 { "VALXCPT" },
588};
589
590#endif
591
592#if 0
593static const struct
594{
595 uint8_t cbPrefixes;
596 uint8_t abPrefixes[14];
597 uint16_t fEffective;
598} g_aPrefixVariations[] =
599{
600 { 0, { 0x00 }, BS3CG1_PF_NONE },
601
602 { 1, { P_OZ }, BS3CG1_PF_OZ },
603 { 1, { P_CS }, BS3CG1_PF_CS },
604 { 1, { P_DS }, BS3CG1_PF_DS },
605 { 1, { P_ES }, BS3CG1_PF_ES },
606 { 1, { P_FS }, BS3CG1_PF_FS },
607 { 1, { P_GS }, BS3CG1_PF_GS },
608 { 1, { P_SS }, BS3CG1_PF_SS },
609 { 1, { P_LK }, BS3CG1_PF_LK },
610
611 { 2, { P_CS, P_OZ, }, BS3CG1_PF_CS | BS3CFG1_PF_OZ },
612 { 2, { P_DS, P_OZ, }, BS3CG1_PF_DS | BS3CFG1_PF_OZ },
613 { 2, { P_ES, P_OZ, }, BS3CG1_PF_ES | BS3CFG1_PF_OZ },
614 { 2, { P_FS, P_OZ, }, BS3CG1_PF_FS | BS3CFG1_PF_OZ },
615 { 2, { P_GS, P_OZ, }, BS3CG1_PF_GS | BS3CFG1_PF_OZ },
616 { 2, { P_GS, P_OZ, }, BS3CG1_PF_SS | BS3CFG1_PF_OZ },
617 { 2, { P_SS, P_OZ, }, BS3CG1_PF_SS | BS3CFG1_PF_OZ },
618
619 { 2, { P_OZ, P_CS, }, BS3CG1_PF_CS | BS3CFG1_PF_OZ },
620 { 2, { P_OZ, P_DS, }, BS3CG1_PF_DS | BS3CFG1_PF_OZ },
621 { 2, { P_OZ, P_ES, }, BS3CG1_PF_ES | BS3CFG1_PF_OZ },
622 { 2, { P_OZ, P_FS, }, BS3CG1_PF_FS | BS3CFG1_PF_OZ },
623 { 2, { P_OZ, P_GS, }, BS3CG1_PF_GS | BS3CFG1_PF_OZ },
624 { 2, { P_OZ, P_GS, }, BS3CG1_PF_SS | BS3CFG1_PF_OZ },
625 { 2, { P_OZ, P_SS, }, BS3CG1_PF_SS | BS3CFG1_PF_OZ },
626};
627
628static const uint16_t g_afPfxKindToIgnoredFlags[BS3CGPFXKIND_END] =
629{
630 /* [BS3CGPFXKIND_INVALID] = */ UINT16_MAX,
631 /* [BS3CGPFXKIND_MODRM] = */ 0,
632 /* [BS3CGPFXKIND_MODRM_NO_OP_SIZES] = */ BS3CG1_PF_OZ | BS3CG1_PF_W,
633};
634
635#endif
636
637
638DECLINLINE(unsigned) Bs3Cg1InsertOpcodes(PBS3CG1STATE pThis, unsigned offDst)
639{
640 switch (pThis->cbOpcodes)
641 {
642 case 4: pThis->abCurInstr[offDst + 3] = pThis->abOpcodes[3];
643 case 3: pThis->abCurInstr[offDst + 2] = pThis->abOpcodes[2];
644 case 2: pThis->abCurInstr[offDst + 1] = pThis->abOpcodes[1];
645 case 1: pThis->abCurInstr[offDst] = pThis->abOpcodes[0];
646 return offDst + pThis->cbOpcodes;
647
648 default:
649 BS3_ASSERT(0);
650 return 0;
651 }
652}
653
654
655/**
656 * Cleans up state and context changes made by the encoder.
657 *
658 * @param pThis The state.
659 */
660static void Bs3Cg1EncodeCleanup(PBS3CG1STATE pThis)
661{
662 /* Restore the DS registers in the contexts. */
663 unsigned iRing = 4;
664 while (iRing-- > 0)
665 pThis->aInitialCtxs[iRing].ds = pThis->aSavedSegRegs[iRing].ds;
666
667 switch (pThis->enmEncoding)
668 {
669 /* Most encodings currently doesn't need any special cleaning up. */
670 default:
671 return;
672 }
673}
674
675
676static unsigned Bs3Cfg1EncodeMemMod0Disp(PBS3CG1STATE pThis, bool fAddrOverride, unsigned off, uint8_t iReg,
677 uint8_t cbOp, BS3CG1OPLOC enmLocation)
678{
679 pThis->aOperands[pThis->iRmOp].idxField = BS3CG1DST_INVALID;
680 pThis->aOperands[pThis->iRmOp].enmLocation = enmLocation;
681 pThis->aOperands[pThis->iRmOp].cbOp = cbOp;
682 pThis->aOperands[pThis->iRmOp].off = cbOp;
683
684 if ( BS3_MODE_IS_16BIT_CODE(pThis->bMode)
685 || (fAddrOverride && BS3_MODE_IS_32BIT_CODE(pThis->bMode)) )
686 {
687 /*
688 * 16-bit code doing 16-bit or 32-bit addressing,
689 * or 32-bit code doing 16-bit addressing.
690 */
691 unsigned iRing = 4;
692 if (BS3_MODE_IS_RM_OR_V86(pThis->bMode))
693 while (iRing-- > 0)
694 pThis->aInitialCtxs[iRing].ds = pThis->DataPgFar.sel;
695 else
696 while (iRing-- > 0)
697 pThis->aInitialCtxs[iRing].ds = pThis->DataPgFar.sel | iRing;
698 if (!fAddrOverride || BS3_MODE_IS_32BIT_CODE(pThis->bMode))
699 {
700 pThis->abCurInstr[off++] = X86_MODRM_MAKE(0, iReg, 6 /*disp16*/);
701 *(uint16_t *)&pThis->abCurInstr[off] = pThis->DataPgFar.off + X86_PAGE_SIZE - cbOp;
702 off += 2;
703 }
704 else
705 {
706 pThis->abCurInstr[off++] = X86_MODRM_MAKE(0, iReg, 5 /*disp32*/);
707 *(uint32_t *)&pThis->abCurInstr[off] = pThis->DataPgFar.off + X86_PAGE_SIZE - cbOp;
708 off += 4;
709 }
710 }
711 else
712 {
713 /*
714 * 32-bit code doing 32-bit addressing,
715 * or 64-bit code doing either 64-bit or 32-bit addressing.
716 */
717 pThis->abCurInstr[off++] = X86_MODRM_MAKE(0, iReg, 5 /*disp32*/);
718 *(uint32_t *)&pThis->abCurInstr[off] = BS3_FP_OFF(pThis->pbDataPg) + X86_PAGE_SIZE - cbOp;
719
720 /* In 64-bit mode we always have a rip relative encoding regardless of fAddrOverride. */
721 if (BS3_MODE_IS_64BIT_CODE(pThis->bMode))
722 *(uint32_t *)&pThis->abCurInstr[off] -= BS3_FP_OFF(&pThis->pbCodePg[X86_PAGE_SIZE]);
723 off += 4;
724 }
725
726 return off;
727}
728
729
730/**
731 * Encodes the next instruction.
732 *
733 * @returns Next iEncoding value. Returns @a iEncoding unchanged to indicate
734 * that there are no more encodings to test.
735 * @param pThis The state.
736 * @param iEncoding The encoding to produce. Meaning is specific to each
737 * BS3CG1ENC_XXX value and should be considered internal.
738 */
739static unsigned Bs3Cg1EncodeNext(PBS3CG1STATE pThis, unsigned iEncoding)
740{
741 unsigned off;
742 unsigned cbOp;
743 switch (pThis->enmEncoding)
744 {
745 case BS3CG1ENC_MODRM_Eb_Gb:
746 /* Start by reg,reg encoding. */
747 if (iEncoding == 0)
748 {
749 off = Bs3Cg1InsertOpcodes(pThis, 0);
750 pThis->abCurInstr[off++] = X86_MODRM_MAKE(3, X86_GREG_xAX, X86_GREG_xCX);
751 pThis->aOperands[pThis->iRegOp].idxField = BS3CG1DST_AL;
752 pThis->aOperands[pThis->iRmOp ].idxField = BS3CG1DST_CL;
753 }
754 else if (iEncoding == 1)
755 {
756 pThis->aOperands[pThis->iRegOp].idxField = BS3CG1DST_CH;
757 off = Bs3Cfg1EncodeMemMod0Disp(pThis, false, Bs3Cg1InsertOpcodes(pThis, 0), X86_GREG_xBP, 1, BS3CG1OPLOC_MEM_RW);
758 }
759 else if (iEncoding == 2 && (g_uBs3CpuDetected & BS3CPU_TYPE_MASK) >= BS3CPU_80386)
760 {
761 pThis->aOperands[pThis->iRegOp].idxField = BS3CG1DST_BH;
762 pThis->abCurInstr[0] = P_AZ;
763 off = Bs3Cfg1EncodeMemMod0Disp(pThis, true, Bs3Cg1InsertOpcodes(pThis, 1), X86_GREG_xDI, 1, BS3CG1OPLOC_MEM_RW);
764 }
765 else
766 break;
767 pThis->cbCurInstr = off;
768 iEncoding++;
769 break;
770
771 case BS3CG1ENC_MODRM_Gb_Eb:
772 /* Start by reg,reg encoding. */
773 if (iEncoding == 0)
774 {
775 off = Bs3Cg1InsertOpcodes(pThis, 0);
776 pThis->abCurInstr[off++] = X86_MODRM_MAKE(3, X86_GREG_xAX, X86_GREG_xCX);
777 pThis->aOperands[pThis->iRegOp].idxField = BS3CG1DST_AL;
778 pThis->aOperands[pThis->iRmOp ].idxField = BS3CG1DST_CL;
779 }
780 else if (iEncoding == 1)
781 {
782 pThis->aOperands[pThis->iRegOp].idxField = BS3CG1DST_CH;
783 off = Bs3Cfg1EncodeMemMod0Disp(pThis, false, Bs3Cg1InsertOpcodes(pThis, 0), X86_GREG_xBP, 1, BS3CG1OPLOC_MEM);
784 }
785 else if (iEncoding == 2 && (g_uBs3CpuDetected & BS3CPU_TYPE_MASK) >= BS3CPU_80386)
786 {
787 pThis->aOperands[pThis->iRegOp].idxField = BS3CG1DST_BH;
788 pThis->abCurInstr[0] = P_AZ;
789 off = Bs3Cfg1EncodeMemMod0Disp(pThis, true, Bs3Cg1InsertOpcodes(pThis, 1), X86_GREG_xDI, 1, BS3CG1OPLOC_MEM);
790 }
791 else
792 break;
793 pThis->cbCurInstr = off;
794 iEncoding++;
795 break;
796
797 case BS3CG1ENC_MODRM_Gv_Ev:
798 case BS3CG1ENC_MODRM_Ev_Gv:
799 if (iEncoding == 0)
800 {
801 cbOp = BS3_MODE_IS_16BIT_CODE(pThis->bMode) ? 2 : 4;
802 off = Bs3Cg1InsertOpcodes(pThis, 0);
803 pThis->abCurInstr[off++] = X86_MODRM_MAKE(3, X86_GREG_xBX, X86_GREG_xDX);
804 pThis->aOperands[pThis->iRegOp].idxField = BS3CG1DST_OZ_RBX;
805 pThis->aOperands[pThis->iRmOp ].idxField = BS3CG1DST_OZ_RDX;
806 }
807 else if (iEncoding == 1)
808 {
809 cbOp = BS3_MODE_IS_16BIT_CODE(pThis->bMode) ? 2 : 4;
810 pThis->aOperands[pThis->iRegOp].idxField = BS3CG1DST_OZ_RBP;
811 off = Bs3Cfg1EncodeMemMod0Disp(pThis, false, Bs3Cg1InsertOpcodes(pThis, 0), X86_GREG_xBP, cbOp,
812 pThis->enmEncoding == BS3CG1ENC_MODRM_Gv_Ev ? BS3CG1OPLOC_MEM : BS3CG1OPLOC_MEM_RW);
813 }
814 else if (iEncoding == 2 && (g_uBs3CpuDetected & BS3CPU_TYPE_MASK) >= BS3CPU_80386)
815 {
816 cbOp = BS3_MODE_IS_16BIT_CODE(pThis->bMode) ? 4 : 2;
817 pThis->abCurInstr[0] = P_OZ;
818 off = Bs3Cg1InsertOpcodes(pThis, 1);
819 pThis->abCurInstr[off++] = X86_MODRM_MAKE(3, X86_GREG_xBX, X86_GREG_xDX);
820 pThis->aOperands[pThis->iRegOp].idxField = BS3CG1DST_OZ_RBX;
821 pThis->aOperands[pThis->iRmOp ].idxField = BS3CG1DST_OZ_RDX;
822 pThis->aOperands[pThis->iRmOp ].enmLocation = BS3CG1OPLOC_CTX;
823 }
824 else if (iEncoding == 3)
825 {
826 cbOp = BS3_MODE_IS_16BIT_CODE(pThis->bMode) ? 4 : 2;
827 pThis->aOperands[pThis->iRegOp].idxField = BS3CG1DST_OZ_RSI;
828 pThis->abCurInstr[0] = P_OZ;
829 off = Bs3Cfg1EncodeMemMod0Disp(pThis, false, Bs3Cg1InsertOpcodes(pThis, 1), X86_GREG_xSI, cbOp,
830 pThis->enmEncoding == BS3CG1ENC_MODRM_Gv_Ev ? BS3CG1OPLOC_MEM : BS3CG1OPLOC_MEM_RW);
831 }
832 else if (iEncoding == 4)
833 {
834 cbOp = BS3_MODE_IS_16BIT_CODE(pThis->bMode) ? 2 : 4;
835 pThis->aOperands[pThis->iRegOp].idxField = BS3CG1DST_OZ_RDI;
836 pThis->abCurInstr[0] = P_AZ;
837 off = Bs3Cfg1EncodeMemMod0Disp(pThis, true, Bs3Cg1InsertOpcodes(pThis, 1), X86_GREG_xDI, cbOp,
838 pThis->enmEncoding == BS3CG1ENC_MODRM_Gv_Ev ? BS3CG1OPLOC_MEM : BS3CG1OPLOC_MEM_RW);
839 }
840 else if (iEncoding == 5)
841 {
842 cbOp = BS3_MODE_IS_16BIT_CODE(pThis->bMode) ? 4 : 2;
843 pThis->aOperands[pThis->iRegOp].idxField = BS3CG1DST_OZ_RSI;
844 pThis->abCurInstr[0] = P_OZ;
845 pThis->abCurInstr[1] = P_AZ;
846 off = Bs3Cfg1EncodeMemMod0Disp(pThis, true, Bs3Cg1InsertOpcodes(pThis, 2), X86_GREG_xSI, cbOp,
847 pThis->enmEncoding == BS3CG1ENC_MODRM_Gv_Ev ? BS3CG1OPLOC_MEM : BS3CG1OPLOC_MEM_RW);
848 }
849 else if (iEncoding == 6 && BS3_MODE_IS_64BIT_CODE(pThis->bMode))
850 {
851 cbOp = 8;
852 pThis->abCurInstr[0] = REX_W___;
853 off = Bs3Cg1InsertOpcodes(pThis, 1);
854 pThis->abCurInstr[off++] = X86_MODRM_MAKE(3, X86_GREG_xBX, X86_GREG_xDX);
855 pThis->aOperands[pThis->iRegOp].idxField = BS3CG1DST_RBX;
856 pThis->aOperands[pThis->iRmOp ].idxField = BS3CG1DST_RDX;
857 pThis->aOperands[pThis->iRmOp ].enmLocation = BS3CG1OPLOC_CTX;
858 }
859 else
860 break;
861 pThis->aOperands[0].cbOp = cbOp;
862 pThis->aOperands[1].cbOp = cbOp;
863 pThis->cbOperand = cbOp;
864 pThis->cbCurInstr = off;
865 iEncoding++;
866 break;
867
868 case BS3CG1ENC_MODRM_Gv_Ma:
869 cbOp = BS3_MODE_IS_16BIT_CODE(pThis->bMode) ? 2 : 4;
870 if (iEncoding == 0)
871 {
872 pThis->aOperands[pThis->iRegOp].idxField = BS3CG1DST_OZ_RBP;
873 off = Bs3Cfg1EncodeMemMod0Disp(pThis, false, Bs3Cg1InsertOpcodes(pThis, 0), X86_GREG_xBP, cbOp * 2, BS3CG1OPLOC_MEM);
874 }
875 else if (iEncoding == 1 && (g_uBs3CpuDetected & BS3CPU_TYPE_MASK) >= BS3CPU_80386)
876 {
877 cbOp = cbOp == 2 ? 4 : 2;
878 pThis->aOperands[pThis->iRegOp].idxField = BS3CG1DST_OZ_RBP;
879 pThis->abCurInstr[0] = P_OZ;
880 off = Bs3Cfg1EncodeMemMod0Disp(pThis, false, Bs3Cg1InsertOpcodes(pThis, 1), X86_GREG_xBP, cbOp * 2, BS3CG1OPLOC_MEM);
881 }
882 else if (iEncoding == 2)
883 {
884 pThis->aOperands[pThis->iRegOp].idxField = BS3CG1DST_OZ_RBP;
885 pThis->abCurInstr[0] = P_AZ;
886 off = Bs3Cfg1EncodeMemMod0Disp(pThis, true, Bs3Cg1InsertOpcodes(pThis, 1), X86_GREG_xBP, cbOp * 2, BS3CG1OPLOC_MEM);
887 }
888 else if (iEncoding == 3)
889 {
890 cbOp = cbOp == 2 ? 4 : 2;
891 pThis->aOperands[pThis->iRegOp].idxField = BS3CG1DST_OZ_RBP;
892 pThis->abCurInstr[0] = P_AZ;
893 pThis->abCurInstr[1] = P_OZ;
894 off = Bs3Cfg1EncodeMemMod0Disp(pThis, true, Bs3Cg1InsertOpcodes(pThis, 2), X86_GREG_xBP, cbOp * 2, BS3CG1OPLOC_MEM);
895 }
896 else
897 break;
898 pThis->aOperands[pThis->iRegOp].cbOp = cbOp;
899 pThis->cbOperand = cbOp;
900 pThis->cbCurInstr = off;
901 iEncoding++;
902 break;
903
904 case BS3CG1ENC_FIXED:
905 if (iEncoding == 0)
906 {
907 off = Bs3Cg1InsertOpcodes(pThis, 0);
908 pThis->cbCurInstr = off;
909 iEncoding++;
910 }
911 break;
912
913 case BS3CG1ENC_FIXED_AL_Ib:
914 if (iEncoding == 0)
915 {
916 off = Bs3Cg1InsertOpcodes(pThis, 0);
917 pThis->aOperands[1].off = (uint8_t)off;
918 pThis->abCurInstr[off++] = 0xff;
919 pThis->cbCurInstr = off;
920 iEncoding++;
921 }
922 break;
923
924 case BS3CG1ENC_FIXED_rAX_Iz:
925 if (iEncoding == 0)
926 {
927 off = Bs3Cg1InsertOpcodes(pThis, 0);
928 pThis->aOperands[1].off = (uint8_t)off;
929 if (BS3_MODE_IS_16BIT_CODE(pThis->bMode))
930 {
931 *(uint16_t *)&pThis->abCurInstr[off] = UINT16_MAX;
932 off += 2;
933 pThis->aOperands[0].cbOp = 2;
934 pThis->aOperands[1].cbOp = 2;
935 pThis->cbOperand = 2;
936 }
937 else
938 {
939 *(uint32_t *)&pThis->abCurInstr[off] = UINT32_MAX;
940 off += 4;
941 pThis->aOperands[0].cbOp = 4;
942 pThis->aOperands[1].cbOp = 4;
943 pThis->cbOperand = 4;
944 }
945 }
946 else if (iEncoding == 1 && (g_uBs3CpuDetected & BS3CPU_TYPE_MASK) >= BS3CPU_80386)
947 {
948 pThis->abCurInstr[0] = P_OZ;
949 off = Bs3Cg1InsertOpcodes(pThis, 1);
950 pThis->aOperands[1].off = (uint8_t)off;
951 if (!BS3_MODE_IS_16BIT_CODE(pThis->bMode))
952 {
953 *(uint16_t *)&pThis->abCurInstr[off] = UINT16_MAX;
954 off += 2;
955 pThis->aOperands[0].cbOp = 2;
956 pThis->aOperands[1].cbOp = 2;
957 pThis->cbOperand = 2;
958 }
959 else
960 {
961 *(uint32_t *)&pThis->abCurInstr[off] = UINT32_MAX;
962 off += 4;
963 pThis->aOperands[0].cbOp = 4;
964 pThis->aOperands[1].cbOp = 4;
965 pThis->cbOperand = 4;
966 }
967 }
968 else if (iEncoding == 2 && BS3_MODE_IS_64BIT_CODE(pThis->bMode))
969 {
970 pThis->abCurInstr[0] = REX_W___;
971 off = Bs3Cg1InsertOpcodes(pThis, 1);
972 pThis->aOperands[1].off = (uint8_t)off;
973 *(uint32_t *)&pThis->abCurInstr[off] = UINT32_MAX;
974 off += 4;
975 pThis->aOperands[0].cbOp = 8;
976 pThis->aOperands[1].cbOp = 4;
977 pThis->cbOperand = 8;
978 }
979 else
980 break;
981 pThis->cbCurInstr = off;
982 iEncoding++;
983 break;
984
985 default:
986 Bs3TestFailedF("Internal error! BS3CG1ENC_XXX = %u not implemented", pThis->enmEncoding);
987 break;
988 }
989
990
991 return iEncoding;
992}
993
994
995/**
996 * Prepares doing instruction encodings.
997 *
998 * This is in part specific to how the instruction is encoded, but generally it
999 * sets up basic operand values that doesn't change (much) when Bs3Cg1EncodeNext
1000 * is called from within the loop.
1001 *
1002 * @returns Success indicator (true/false).
1003 * @param pThis The state.
1004 */
1005static bool Bs3Cg1EncodePrep(PBS3CG1STATE pThis)
1006{
1007 unsigned iRing = 4;
1008 while (iRing-- > 0)
1009 pThis->aSavedSegRegs[iRing].ds = pThis->aInitialCtxs[iRing].ds;
1010
1011 pThis->iRmOp = RT_ELEMENTS(pThis->aOperands) - 1;
1012 pThis->iRegOp = RT_ELEMENTS(pThis->aOperands) - 1;
1013 pThis->fSameRingNotOkay = false;
1014
1015 switch (pThis->enmEncoding)
1016 {
1017 case BS3CG1ENC_MODRM_Eb_Gb:
1018 pThis->iRmOp = 0;
1019 pThis->iRegOp = 1;
1020 pThis->aOperands[0].cbOp = 1;
1021 pThis->aOperands[1].cbOp = 1;
1022 pThis->aOperands[0].enmLocation = BS3CG1OPLOC_CTX;
1023 pThis->aOperands[1].enmLocation = BS3CG1OPLOC_CTX;
1024 break;
1025
1026 case BS3CG1ENC_MODRM_Ev_Gv:
1027 pThis->iRmOp = 0;
1028 pThis->iRegOp = 1;
1029 pThis->aOperands[0].cbOp = 2;
1030 pThis->aOperands[1].cbOp = 2;
1031 pThis->aOperands[0].enmLocation = BS3CG1OPLOC_CTX;
1032 pThis->aOperands[1].enmLocation = BS3CG1OPLOC_CTX;
1033 break;
1034
1035 case BS3CG1ENC_MODRM_Gb_Eb:
1036 pThis->iRmOp = 1;
1037 pThis->iRegOp = 0;
1038 pThis->aOperands[0].cbOp = 1;
1039 pThis->aOperands[1].cbOp = 1;
1040 pThis->aOperands[0].enmLocation = BS3CG1OPLOC_CTX;
1041 pThis->aOperands[1].enmLocation = BS3CG1OPLOC_CTX;
1042 break;
1043
1044 case BS3CG1ENC_MODRM_Gv_Ev:
1045 pThis->iRmOp = 1;
1046 pThis->iRegOp = 0;
1047 pThis->aOperands[0].cbOp = 2;
1048 pThis->aOperands[1].cbOp = 2;
1049 pThis->aOperands[0].enmLocation = BS3CG1OPLOC_CTX;
1050 pThis->aOperands[1].enmLocation = BS3CG1OPLOC_CTX;
1051 break;
1052
1053 case BS3CG1ENC_MODRM_Gv_Ma:
1054 pThis->iRmOp = 1;
1055 pThis->iRegOp = 0;
1056 pThis->aOperands[0].cbOp = 2;
1057 pThis->aOperands[1].cbOp = 4;
1058 pThis->aOperands[0].enmLocation = BS3CG1OPLOC_CTX;
1059 pThis->aOperands[1].enmLocation = BS3CG1OPLOC_MEM;
1060 pThis->aOperands[1].idxField = BS3CG1DST_INVALID;
1061 break;
1062
1063 case BS3CG1ENC_FIXED:
1064 /* nothing to do here */
1065 break;
1066
1067 case BS3CG1ENC_FIXED_AL_Ib:
1068 pThis->aOperands[0].cbOp = 1;
1069 pThis->aOperands[1].cbOp = 1;
1070 pThis->aOperands[0].enmLocation = BS3CG1OPLOC_CTX;
1071 pThis->aOperands[1].enmLocation = BS3CG1OPLOC_IMM;
1072 pThis->aOperands[0].idxField = BS3CG1DST_AL;
1073 pThis->aOperands[1].idxField = BS3CG1DST_INVALID;
1074 break;
1075
1076 case BS3CG1ENC_FIXED_rAX_Iz:
1077 pThis->aOperands[0].cbOp = 2;
1078 pThis->aOperands[1].cbOp = 2;
1079 pThis->aOperands[0].enmLocation = BS3CG1OPLOC_CTX;
1080 pThis->aOperands[1].enmLocation = BS3CG1OPLOC_IMM;
1081 pThis->aOperands[0].idxField = BS3CG1DST_OZ_RAX;
1082 pThis->aOperands[1].idxField = BS3CG1DST_INVALID;
1083 break;
1084
1085 default:
1086 return Bs3TestFailedF("Invalid/unimplemented enmEncoding for instruction #%RU32 (%.*s): %d",
1087 pThis->iInstr, pThis->cchMnemonic, pThis->pchMnemonic, pThis->enmEncoding);
1088 }
1089 return true;
1090}
1091
1092
1093static bool Bs3Cg3SetupAvx(PBS3CG1STATE pThis)
1094{
1095 return true;
1096}
1097
1098
1099/**
1100 * Sets up SSE.
1101 *
1102 * @returns true (if successful, false if not and the SSE instructions ends up
1103 * being invalid).
1104 * @param pThis The state.
1105 */
1106static bool Bs3Cg3SetupSse(PBS3CG1STATE pThis)
1107{
1108 ASMSetCR4(ASMGetCR4() | X86_CR4_OSFXSR | X86_CR4_OSXMMEEXCPT);
1109 return true;
1110}
1111
1112
1113/**
1114 * Check if the instruction is supported by the CPU, possibly making state
1115 * adjustments to enable support for it.
1116 *
1117 * @returns true if supported, false if not.
1118 * @param pThis The state.
1119 */
1120static bool Bs3Cg1CpuTestAndEnable(PBS3CG1STATE pThis)
1121{
1122 uint32_t fEax;
1123 uint32_t fEbx;
1124 uint32_t fEcx;
1125 uint32_t fEdx;
1126 ASMCpuIdExSlow(1, 0, 0, 0, NULL, NULL, &fEcx, &fEdx);
1127
1128 if ( (pThis->fFlags & BS3CG1INSTR_F_INVALID_64BIT)
1129 && BS3_MODE_IS_64BIT_CODE(pThis->bMode))
1130 return false;
1131
1132 switch (pThis->enmCpuTest)
1133 {
1134 case BS3CG1CPU_ANY:
1135 return true;
1136
1137 case BS3CG1CPU_GE_80186:
1138 if ((g_uBs3CpuDetected & BS3CPU_TYPE_MASK) >= BS3CPU_80186)
1139 return true;
1140 return false;
1141
1142 case BS3CG1CPU_GE_80286:
1143 if ((g_uBs3CpuDetected & BS3CPU_TYPE_MASK) >= BS3CPU_80286)
1144 return true;
1145 return false;
1146
1147 case BS3CG1CPU_GE_80386:
1148 if ((g_uBs3CpuDetected & BS3CPU_TYPE_MASK) >= BS3CPU_80386)
1149 return true;
1150 return false;
1151
1152 case BS3CG1CPU_GE_80486:
1153 if ((g_uBs3CpuDetected & BS3CPU_TYPE_MASK) >= BS3CPU_80486)
1154 return true;
1155 return false;
1156
1157 case BS3CG1CPU_GE_Pentium:
1158 if ((g_uBs3CpuDetected & BS3CPU_TYPE_MASK) >= BS3CPU_Pentium)
1159 return true;
1160 return false;
1161
1162 case BS3CG1CPU_SSE:
1163 case BS3CG1CPU_SSE2:
1164 case BS3CG1CPU_SSE3:
1165 case BS3CG1CPU_AVX:
1166 if (g_uBs3CpuDetected & BS3CPU_F_CPUID)
1167 {
1168 ASMCpuIdExSlow(1, 0, 0, 0, NULL, NULL, &fEcx, &fEdx);
1169 switch (pThis->enmCpuTest)
1170 {
1171 case BS3CG1CPU_SSE:
1172 if (fEdx & X86_CPUID_FEATURE_EDX_SSE)
1173 return Bs3Cg3SetupSse(pThis);
1174 return false;
1175 case BS3CG1CPU_SSE2:
1176 if (fEdx & X86_CPUID_FEATURE_EDX_SSE2)
1177 return Bs3Cg3SetupSse(pThis);
1178 return false;
1179 case BS3CG1CPU_SSE3:
1180 if (fEcx & X86_CPUID_FEATURE_ECX_SSE3)
1181 return Bs3Cg3SetupSse(pThis);
1182 return false;
1183 case BS3CG1CPU_AVX:
1184 if (fEcx & X86_CPUID_FEATURE_ECX_AVX)
1185 return Bs3Cg3SetupAvx(pThis);
1186 return false;
1187 default: BS3_ASSERT(0); /* impossible */
1188 }
1189 }
1190 return false;
1191
1192 case BS3CG1CPU_AVX2:
1193 if (g_uBs3CpuDetected & BS3CPU_F_CPUID)
1194 {
1195 ASMCpuIdExSlow(7, 0, 0/*leaf*/, 0, &fEax, &fEbx, &fEcx, &fEdx);
1196
1197 switch (pThis->enmCpuTest)
1198 {
1199 case BS3CG1CPU_AVX2:
1200 if (fEbx & X86_CPUID_STEXT_FEATURE_EBX_AVX2)
1201 return Bs3Cg3SetupAvx(pThis);
1202 return false;
1203 default: BS3_ASSERT(0); return false; /* impossible */
1204 }
1205 }
1206 return false;
1207
1208 default:
1209 Bs3TestFailedF("Invalid enmCpuTest value: %d", pThis->enmCpuTest);
1210 return false;
1211 }
1212}
1213
1214
1215
1216/**
1217 * Checks the preconditions for a test.
1218 *
1219 * @returns true if the test be executed, false if not.
1220 * @param pThis The state.
1221 * @param pHdr The test header.
1222 */
1223static bool Bs3Cg1RunSelector(PBS3CG1STATE pThis, PCBS3CG1TESTHDR pHdr)
1224{
1225
1226 uint8_t const BS3_FAR *pbCode = (uint8_t const BS3_FAR *)(pHdr + 1);
1227 unsigned cbLeft = pHdr->cbSelector;
1228 while (cbLeft-- > 0)
1229 {
1230 switch (*pbCode++)
1231 {
1232#define CASE_PRED(a_Pred, a_Expr) \
1233 case ((a_Pred) << BS3CG1SEL_OP_KIND_MASK) | BS3CG1SEL_OP_IS_TRUE: \
1234 if (!(a_Expr)) return false; \
1235 break; \
1236 case ((a_Pred) << BS3CG1SEL_OP_KIND_MASK) | BS3CG1SEL_OP_IS_FALSE: \
1237 if (a_Expr) return false; \
1238 break
1239 CASE_PRED(BS3CG1PRED_SIZE_O16, pThis->cbOperand == 2);
1240 CASE_PRED(BS3CG1PRED_SIZE_O32, pThis->cbOperand == 4);
1241 CASE_PRED(BS3CG1PRED_SIZE_O64, pThis->cbOperand == 8);
1242 CASE_PRED(BS3CG1PRED_RING_0, pThis->uCpl == 0);
1243 CASE_PRED(BS3CG1PRED_RING_1, pThis->uCpl == 1);
1244 CASE_PRED(BS3CG1PRED_RING_2, pThis->uCpl == 2);
1245 CASE_PRED(BS3CG1PRED_RING_3, pThis->uCpl == 3);
1246 CASE_PRED(BS3CG1PRED_RING_0_THRU_2, pThis->uCpl <= 2);
1247 CASE_PRED(BS3CG1PRED_RING_1_THRU_3, pThis->uCpl >= 1);
1248 CASE_PRED(BS3CG1PRED_CODE_64BIT, BS3_MODE_IS_64BIT_CODE(pThis->bMode));
1249 CASE_PRED(BS3CG1PRED_CODE_32BIT, BS3_MODE_IS_32BIT_CODE(pThis->bMode));
1250 CASE_PRED(BS3CG1PRED_CODE_16BIT, BS3_MODE_IS_16BIT_CODE(pThis->bMode));
1251 CASE_PRED(BS3CG1PRED_MODE_REAL, BS3_MODE_IS_RM_SYS(pThis->bMode));
1252 CASE_PRED(BS3CG1PRED_MODE_PROT, BS3_MODE_IS_PM_SYS(pThis->bMode));
1253 CASE_PRED(BS3CG1PRED_MODE_LONG, BS3_MODE_IS_64BIT_SYS(pThis->bMode));
1254 CASE_PRED(BS3CG1PRED_MODE_SMM, false);
1255 CASE_PRED(BS3CG1PRED_MODE_VMX, false);
1256 CASE_PRED(BS3CG1PRED_MODE_SVM, false);
1257 CASE_PRED(BS3CG1PRED_PAGING_ON, BS3_MODE_IS_PAGED(pThis->bMode));
1258 CASE_PRED(BS3CG1PRED_PAGING_OFF, !BS3_MODE_IS_PAGED(pThis->bMode));
1259
1260#undef CASE_PRED
1261 default:
1262 return Bs3TestFailedF("Invalid selector opcode %#x!", pbCode[-1]);
1263 }
1264 }
1265
1266 return true;
1267}
1268
1269
1270/**
1271 * Runs a context modifier program.
1272 *
1273 * @returns Success indicator (true/false).
1274 * @param pThis The state.
1275 * @param pCtx The context.
1276 * @param pHdr The program header.
1277 * @param off The program offset relative to the end of the header.
1278 * @param cb The program size.
1279 * @param pEflCtx The context to take undefined EFLAGS from. (This is NULL
1280 * if we're processing a input context modifier program.)
1281 * @param pbInstr Points to the first instruction byte. For storing
1282 * immediate operands during input context modification.
1283 * NULL for output contexts.
1284 */
1285static bool Bs3Cg1RunContextModifier(PBS3CG1STATE pThis, PBS3REGCTX pCtx, PCBS3CG1TESTHDR pHdr, unsigned off, unsigned cb,
1286 PCBS3REGCTX pEflCtx, uint8_t BS3_FAR *pbInstr)
1287{
1288 uint8_t const BS3_FAR *pbCode = (uint8_t const BS3_FAR *)(pHdr + 1) + off;
1289 int cbLeft = cb;
1290 while (cbLeft-- > 0)
1291 {
1292 /*
1293 * Decode the instruction.
1294 */
1295 uint8_t const bOpcode = *pbCode++;
1296 unsigned cbValue;
1297 unsigned cbDst;
1298 BS3CG1DST idxField;
1299
1300 /* Expand the destiation field (can be escaped). */
1301 switch (bOpcode & BS3CG1_CTXOP_DST_MASK)
1302 {
1303 case BS3CG1_CTXOP_OP1:
1304 idxField = pThis->aOperands[0].idxField;
1305 if (idxField == BS3CG1DST_INVALID)
1306 idxField = BS3CG1DST_OP1;
1307 break;
1308
1309 case BS3CG1_CTXOP_OP2:
1310 idxField = pThis->aOperands[1].idxField;
1311 if (idxField == BS3CG1DST_INVALID)
1312 idxField = BS3CG1DST_OP2;
1313 break;
1314
1315 case BS3CG1_CTXOP_EFL:
1316 idxField = BS3CG1DST_EFL;
1317 break;
1318
1319 case BS3CG1_CTXOP_DST_ESC:
1320 if (cbLeft-- > 0)
1321 {
1322 idxField = (BS3CG1DST)*pbCode++;
1323 if (idxField <= BS3CG1DST_OP4)
1324 {
1325 if (idxField > BS3CG1DST_INVALID)
1326 {
1327 uint8_t idxField2 = pThis->aOperands[idxField - BS3CG1DST_OP1].idxField;
1328 if (idxField2 != BS3CG1DST_INVALID)
1329 idxField = idxField2;
1330 break;
1331 }
1332 }
1333 else if (idxField < BS3CG1DST_END)
1334 break;
1335 return Bs3TestFailedF("Malformed context instruction: idxField=%d", idxField);
1336 }
1337 /* fall thru */
1338 default:
1339 return Bs3TestFailed("Malformed context instruction: Destination");
1340 }
1341
1342
1343 /* Expand value size (can be escaped). */
1344 switch (bOpcode & BS3CG1_CTXOP_SIZE_MASK)
1345 {
1346 case BS3CG1_CTXOP_1_BYTE: cbValue = 1; break;
1347 case BS3CG1_CTXOP_2_BYTES: cbValue = 2; break;
1348 case BS3CG1_CTXOP_4_BYTES: cbValue = 4; break;
1349 case BS3CG1_CTXOP_8_BYTES: cbValue = 8; break;
1350 case BS3CG1_CTXOP_16_BYTES: cbValue = 16; break;
1351 case BS3CG1_CTXOP_32_BYTES: cbValue = 32; break;
1352 case BS3CG1_CTXOP_12_BYTES: cbValue = 12; break;
1353 case BS3CG1_CTXOP_SIZE_ESC:
1354 if (cbLeft-- > 0)
1355 {
1356 cbValue = *pbCode++;
1357 if (cbValue)
1358 break;
1359 }
1360 /* fall thru */
1361 default:
1362 return Bs3TestFailed("Malformed context instruction: size");
1363 }
1364
1365 /* Make sure there is enough instruction bytes for the value. */
1366 if (cbValue <= cbLeft)
1367 { /* likely */ }
1368 else
1369 return Bs3TestFailedF("Malformed context instruction: %u bytes value, %u bytes left", cbValue, cbLeft);
1370
1371 /*
1372 * Do value processing specific to the target field size.
1373 */
1374 cbDst = g_acbBs3Cg1DstFields[idxField];
1375 if (cbDst == BS3CG1DSTSIZE_OPERAND)
1376 cbDst = pThis->aOperands[idxField - BS3CG1DST_OP1].cbOp;
1377 else if (cbDst == BS3CG1DSTSIZE_OPERAND_SIZE_GRP)
1378 cbDst = pThis->cbOperand;
1379 if (cbDst <= 8)
1380 {
1381 unsigned const offField = g_aoffBs3Cg1DstFields[idxField];
1382 BS3PTRUNION PtrField;
1383
1384 /*
1385 * Deal with fields up to 8-byte wide.
1386 */
1387 /* Get the value. */
1388 uint64_t uValue;
1389 if ((bOpcode & BS3CG1_CTXOP_SIGN_EXT))
1390 switch (cbValue)
1391 {
1392 case 1: uValue = *(int8_t const BS3_FAR *)pbCode; break;
1393 case 2: uValue = *(int16_t const BS3_FAR *)pbCode; break;
1394 case 4: uValue = *(int32_t const BS3_FAR *)pbCode; break;
1395 default:
1396 if (cbValue >= 8)
1397 {
1398 uValue = *(uint64_t const BS3_FAR *)pbCode;
1399 break;
1400 }
1401 return Bs3TestFailedF("Malformed context instruction: %u bytes value (%u dst)", cbValue, cbDst);
1402 }
1403 else
1404 switch (cbValue)
1405 {
1406 case 1: uValue = *(uint8_t const BS3_FAR *)pbCode; break;
1407 case 2: uValue = *(uint16_t const BS3_FAR *)pbCode; break;
1408 case 4: uValue = *(uint32_t const BS3_FAR *)pbCode; break;
1409 default:
1410 if (cbValue >= 8)
1411 {
1412 uValue = *(uint64_t const BS3_FAR *)pbCode;
1413 break;
1414 }
1415 return Bs3TestFailedF("Malformed context instruction: %u bytes value (%u dst)", cbValue, cbDst);
1416 }
1417
1418 /* Find the field. */
1419 if (offField < sizeof(BS3REGCTX))
1420 PtrField.pu8 = (uint8_t BS3_FAR *)pCtx + offField;
1421 /* Non-register operands: */
1422 else if ((unsigned)(idxField - BS3CG1DST_OP1) < 4U)
1423 {
1424 unsigned const idxOp = idxField - BS3CG1DST_OP1;
1425
1426 switch (pThis->aOperands[idxOp].enmLocation)
1427 {
1428 case BS3CG1OPLOC_IMM:
1429 if (pbInstr)
1430 PtrField.pu8 = &pbInstr[pThis->aOperands[idxOp].off];
1431 else
1432 return Bs3TestFailedF("Immediate operand referenced in output context!");
1433 break;
1434
1435 case BS3CG1OPLOC_MEM:
1436 if (!pbInstr)
1437 return Bs3TestFailedF("Read only operand specified in output!");
1438 PtrField.pu8 = &pThis->pbDataPg[X86_PAGE_SIZE - pThis->aOperands[idxOp].off];
1439 break;
1440
1441 case BS3CG1OPLOC_MEM_RW:
1442 if (pbInstr)
1443 PtrField.pu8 = &pThis->pbDataPg[X86_PAGE_SIZE - pThis->aOperands[idxOp].off];
1444 else
1445 PtrField.pu8 = pThis->MemOp.ab;
1446 break;
1447
1448 default:
1449 return Bs3TestFailedF("Internal error: cbDst=%u idxField=%d (%d) offField=%#x: enmLocation=%u off=%#x idxField=%u",
1450 cbDst, idxField, idxOp, offField, pThis->aOperands[idxOp].enmLocation,
1451 pThis->aOperands[idxOp].off, pThis->aOperands[idxOp].idxField);
1452 }
1453 }
1454 /* Special field: Copying in undefined EFLAGS from the result context. */
1455 else if (idxField == BS3CG1DST_EFL_UNDEF)
1456 {
1457 if (!pEflCtx || (bOpcode & BS3CG1_CTXOP_OPERATOR_MASK) != BS3CG1_CTXOP_ASSIGN)
1458 return Bs3TestFailed("Invalid BS3CG1DST_EFL_UNDEF usage");
1459 PtrField.pu32 = &pCtx->rflags.u32;
1460 uValue = (*PtrField.pu32 & ~(uint32_t)uValue) | (pEflCtx->rflags.u32 & (uint32_t)uValue);
1461 }
1462 /* Special field: Expected value (in/result) exception. */
1463 else if (idxField == BS3CG1DST_VALUE_XCPT)
1464 {
1465 if (!pEflCtx || (bOpcode & BS3CG1_CTXOP_OPERATOR_MASK) != BS3CG1_CTXOP_ASSIGN || cbDst != 1)
1466 return Bs3TestFailed("Invalid BS3CG1DST_VALUE_XCPT usage");
1467 PtrField.pu8 = &pThis->bValueXcpt;
1468 }
1469 /// @todo else if (idxField <= BS3CG1DST_OP4)
1470 /// @todo {
1471 /// @todo
1472 /// @todo }
1473 else
1474 return Bs3TestFailedF("Todo implement me: cbDst=%u idxField=%d offField=%#x", cbDst, idxField, offField);
1475
1476#ifdef BS3CG1_DEBUG_CTX_MOD
1477 {
1478 const char BS3_FAR *pszOp;
1479 switch (bOpcode & BS3CG1_CTXOP_OPERATOR_MASK)
1480 {
1481 case BS3CG1_CTXOP_ASSIGN: pszOp = "="; break;
1482 case BS3CG1_CTXOP_OR: pszOp = "|="; break;
1483 case BS3CG1_CTXOP_AND: pszOp = "&="; break;
1484 case BS3CG1_CTXOP_AND_INV: pszOp = "&~="; break;
1485 }
1486 switch (cbDst)
1487 {
1488 case 1:
1489 BS3CG1_DPRINTF(("dbg: modify %s: %#04RX8 (LB %u) %s %#RX64 (LB %u)\n",
1490 g_aszBs3Cg1DstFields[idxField].sz, *PtrField.pu8, cbDst, pszOp, uValue, cbValue));
1491 break;
1492 case 2:
1493 BS3CG1_DPRINTF(("dbg: modify %s: %#06RX16 (LB %u) %s %#RX64 (LB %u)\n",
1494 g_aszBs3Cg1DstFields[idxField].sz, *PtrField.pu16, cbDst, pszOp, uValue, cbValue));
1495 break;
1496 case 4:
1497 BS3CG1_DPRINTF(("dbg: modify %s: %#010RX32 (LB %u) %s %#RX64 (LB %u)\n",
1498 g_aszBs3Cg1DstFields[idxField].sz, *PtrField.pu32, cbDst, pszOp, uValue, cbValue));
1499 break;
1500 default:
1501 BS3CG1_DPRINTF(("dbg: modify %s: %#018RX64 (LB %u) %s %#RX64 (LB %u)\n",
1502 g_aszBs3Cg1DstFields[idxField].sz, *PtrField.pu64, cbDst, pszOp, uValue, cbValue));
1503 break;
1504 }
1505 }
1506#endif
1507
1508 /* Modify the field. */
1509 switch (cbDst)
1510 {
1511 case 1:
1512 switch (bOpcode & BS3CG1_CTXOP_OPERATOR_MASK)
1513 {
1514 case BS3CG1_CTXOP_ASSIGN: *PtrField.pu8 = (uint8_t)uValue; break;
1515 case BS3CG1_CTXOP_OR: *PtrField.pu8 |= (uint8_t)uValue; break;
1516 case BS3CG1_CTXOP_AND: *PtrField.pu8 &= (uint8_t)uValue; break;
1517 case BS3CG1_CTXOP_AND_INV: *PtrField.pu8 &= ~(uint8_t)uValue; break;
1518 }
1519 break;
1520
1521 case 2:
1522 switch (bOpcode & BS3CG1_CTXOP_OPERATOR_MASK)
1523 {
1524 case BS3CG1_CTXOP_ASSIGN: *PtrField.pu16 = (uint16_t)uValue; break;
1525 case BS3CG1_CTXOP_OR: *PtrField.pu16 |= (uint16_t)uValue; break;
1526 case BS3CG1_CTXOP_AND: *PtrField.pu16 &= (uint16_t)uValue; break;
1527 case BS3CG1_CTXOP_AND_INV: *PtrField.pu16 &= ~(uint16_t)uValue; break;
1528 }
1529 break;
1530
1531 case 4:
1532 switch (bOpcode & BS3CG1_CTXOP_OPERATOR_MASK)
1533 {
1534 case BS3CG1_CTXOP_ASSIGN: *PtrField.pu32 = (uint32_t)uValue; break;
1535 case BS3CG1_CTXOP_OR: *PtrField.pu32 |= (uint32_t)uValue; break;
1536 case BS3CG1_CTXOP_AND: *PtrField.pu32 &= (uint32_t)uValue; break;
1537 case BS3CG1_CTXOP_AND_INV: *PtrField.pu32 &= ~(uint32_t)uValue; break;
1538 }
1539 break;
1540
1541 case 8:
1542 switch (bOpcode & BS3CG1_CTXOP_OPERATOR_MASK)
1543 {
1544 case BS3CG1_CTXOP_ASSIGN: *PtrField.pu64 = (uint64_t)uValue; break;
1545 case BS3CG1_CTXOP_OR: *PtrField.pu64 |= (uint64_t)uValue; break;
1546 case BS3CG1_CTXOP_AND: *PtrField.pu64 &= (uint64_t)uValue; break;
1547 case BS3CG1_CTXOP_AND_INV: *PtrField.pu64 &= ~(uint64_t)uValue; break;
1548 }
1549 break;
1550
1551 default:
1552 return Bs3TestFailedF("Malformed context instruction: cbDst=%u, expected 1, 2, 4, or 8", cbDst);
1553 }
1554
1555#ifdef BS3CG1_DEBUG_CTX_MOD
1556 switch (cbDst)
1557 {
1558 case 1: BS3CG1_DPRINTF(("dbg: --> %s: %#04RX8\n", g_aszBs3Cg1DstFields[idxField].sz, *PtrField.pu8)); break;
1559 case 2: BS3CG1_DPRINTF(("dbg: --> %s: %#06RX16\n", g_aszBs3Cg1DstFields[idxField].sz, *PtrField.pu16)); break;
1560 case 4: BS3CG1_DPRINTF(("dbg: --> %s: %#010RX32\n", g_aszBs3Cg1DstFields[idxField].sz, *PtrField.pu32)); break;
1561 default: BS3CG1_DPRINTF(("dbg: --> %s: %#018RX64\n", g_aszBs3Cg1DstFields[idxField].sz, *PtrField.pu64)); break;
1562 }
1563#endif
1564
1565 }
1566 /*
1567 * Deal with larger field (FPU, SSE, AVX, ...).
1568 */
1569 else
1570 return Bs3TestFailedF("TODO: Implement me: cbDst=%u idxField=%d", cbDst, idxField);
1571
1572 /*
1573 * Advance to the next instruction.
1574 */
1575 pbCode += cbValue;
1576 cbLeft -= cbValue;
1577 }
1578
1579 return true;
1580}
1581
1582
1583/**
1584 * Checks the result of a run.
1585 *
1586 * @returns true if successful, false if not.
1587 * @param pThis The state.
1588 * @param fInvalidInstr Whether this is an invalid instruction.
1589 * @param bTestXcptExpected The exception causing the test code to stop
1590 * executing.
1591 * @param iEncoding For error reporting.
1592 */
1593static bool Bs3Cg1CheckResult(PBS3CG1STATE pThis, bool fInvalidInstr, uint8_t bTestXcptExpected, unsigned iEncoding)
1594{
1595 unsigned iOperand;
1596
1597 /*
1598 * Check the exception state first.
1599 */
1600 uint8_t bExpectedXcpt;
1601 uint8_t cbAdjustPc;
1602 if (!fInvalidInstr)
1603 {
1604 bExpectedXcpt = pThis->bValueXcpt;
1605 if (bExpectedXcpt == UINT8_MAX)
1606 {
1607 cbAdjustPc = pThis->cbCurInstr;
1608 bExpectedXcpt = bTestXcptExpected;
1609 if (bTestXcptExpected == X86_XCPT_PF)
1610 pThis->Ctx.cr2.u = pThis->uCodePgFlat + X86_PAGE_SIZE;
1611 }
1612 else
1613 cbAdjustPc = 0;
1614 }
1615 else
1616 {
1617 cbAdjustPc = 0;
1618 bExpectedXcpt = bTestXcptExpected;
1619 }
1620 if (RT_LIKELY( pThis->TrapFrame.bXcpt == bExpectedXcpt
1621 && pThis->TrapFrame.Ctx.rip.u == pThis->Ctx.rip.u + cbAdjustPc))
1622 {
1623 if (RT_LIKELY(Bs3TestCheckRegCtxEx(&pThis->TrapFrame.Ctx, &pThis->Ctx,
1624 cbAdjustPc, 0 /*cbSpAdjust*/, 0 /*fExtraEfl*/,
1625 pThis->pszMode, iEncoding)))
1626 {
1627 bool fOkay = true;
1628 iOperand = pThis->cOperands;
1629 while (iOperand-- > 0)
1630 if (pThis->aOperands[iOperand].enmLocation == BS3CG1OPLOC_MEM_RW)
1631 {
1632 BS3PTRUNION PtrUnion;
1633 PtrUnion.pb = &pThis->pbDataPg[X86_PAGE_SIZE - pThis->aOperands[iOperand].off];
1634 switch (pThis->aOperands[iOperand].cbOp)
1635 {
1636 case 1:
1637 if (*PtrUnion.pu8 == pThis->MemOp.ab[0])
1638 continue;
1639 Bs3TestFailedF("op%u: Wrote %#04RX8, expected %#04RX8", iOperand, *PtrUnion.pu8, pThis->MemOp.ab[0]);
1640 break;
1641 case 2:
1642 if (*PtrUnion.pu16 == pThis->MemOp.au16[0])
1643 continue;
1644 Bs3TestFailedF("op%u: Wrote %#06RX16, expected %#06RX16",
1645 iOperand, *PtrUnion.pu16, pThis->MemOp.au16[0]);
1646 break;
1647 case 4:
1648 if (*PtrUnion.pu32 == pThis->MemOp.au32[0])
1649 continue;
1650 Bs3TestFailedF("op%u: Wrote %#010RX32, expected %#010RX32",
1651 iOperand, *PtrUnion.pu32, pThis->MemOp.au32[0]);
1652 break;
1653 case 8:
1654 if (*PtrUnion.pu64 == pThis->MemOp.au64[0])
1655 continue;
1656 Bs3TestFailedF("op%u: Wrote %#018RX64, expected %#018RX64",
1657 iOperand, *PtrUnion.pu64, pThis->MemOp.au64[0]);
1658 break;
1659 default:
1660 if (Bs3MemCmp(PtrUnion.pb, pThis->MemOp.ab, pThis->aOperands[iOperand].cbOp) == 0)
1661 continue;
1662 Bs3TestFailedF("op%u: Wrote %.*Rhxs, expected %.*Rhxs",
1663 iOperand,
1664 pThis->aOperands[iOperand].cbOp, PtrUnion.pb,
1665 pThis->aOperands[iOperand].cbOp, pThis->MemOp.ab);
1666 break;
1667 }
1668 fOkay = false;
1669 }
1670 if (fOkay)
1671 return true;
1672 }
1673
1674 /*
1675 * Report failure.
1676 */
1677 Bs3TestFailedF("%RU32[%u]: encoding#%u: %.*Rhxs",
1678 pThis->iInstr, pThis->iTest, iEncoding, pThis->cbCurInstr, pThis->abCurInstr);
1679 }
1680 else
1681 Bs3TestFailedF("%RU32[%u]: bXcpt=%#x expected %#x; rip=%RX64 expected %RX64; encoding#%u: %.*Rhxs",
1682 pThis->iInstr, pThis->iTest,
1683 pThis->TrapFrame.bXcpt, bExpectedXcpt,
1684 pThis->TrapFrame.Ctx.rip.u, pThis->Ctx.rip.u + cbAdjustPc,
1685 iEncoding, pThis->cbCurInstr, pThis->abCurInstr);
1686 Bs3TestPrintf("cpl=%u cbOperands=%u\n", pThis->uCpl, pThis->cbOperand);
1687
1688 /* Display memory operands. */
1689 for (iOperand = 0; iOperand < pThis->cOperands; iOperand++)
1690 {
1691 BS3PTRUNION PtrUnion;
1692 switch (pThis->aOperands[iOperand].enmLocation)
1693 {
1694 case BS3CG1OPLOC_CTX:
1695 {
1696 uint8_t idxField = pThis->aOperands[iOperand].idxField;
1697 unsigned offField = g_aoffBs3Cg1DstFields[idxField];
1698 if (offField <= sizeof(BS3REGCTX))
1699 PtrUnion.pb = (uint8_t BS3_FAR *)&pThis->Ctx + offField;
1700 else
1701 {
1702 Bs3TestPrintf("op%u: imm%u: %.*Rhxs\n", iOperand, pThis->aOperands[iOperand].cbOp * 8,
1703 pThis->aOperands[iOperand].cbOp, *PtrUnion.pu64);
1704 break;
1705 }
1706 switch (pThis->aOperands[iOperand].cbOp)
1707 {
1708 case 1: Bs3TestPrintf("op%u: ctx08: %#04RX8\n", iOperand, *PtrUnion.pu8); break;
1709 case 2: Bs3TestPrintf("op%u: ctx16: %#06RX16\n", iOperand, *PtrUnion.pu16); break;
1710 case 4: Bs3TestPrintf("op%u: ctx32: %#010RX32\n", iOperand, *PtrUnion.pu32); break;
1711 case 8: Bs3TestPrintf("op%u: ctx64: %#018RX64\n", iOperand, *PtrUnion.pu64); break;
1712 default:
1713 Bs3TestPrintf("op%u: ctx%u: %.*Rhxs\n", iOperand, pThis->aOperands[iOperand].cbOp * 8,
1714 pThis->aOperands[iOperand].cbOp, *PtrUnion.pu64);
1715 break;
1716 }
1717 break;
1718 }
1719
1720 case BS3CG1OPLOC_IMM:
1721 PtrUnion.pb = &pThis->pbCodePg[pThis->aOperands[iOperand].off];
1722 switch (pThis->aOperands[iOperand].cbOp)
1723 {
1724 case 1: Bs3TestPrintf("op%u: imm08: %#04RX8\n", iOperand, *PtrUnion.pu8); break;
1725 case 2: Bs3TestPrintf("op%u: imm16: %#06RX16\n", iOperand, *PtrUnion.pu16); break;
1726 case 4: Bs3TestPrintf("op%u: imm32: %#010RX32\n", iOperand, *PtrUnion.pu32); break;
1727 case 8: Bs3TestPrintf("op%u: imm64: %#018RX64\n", iOperand, *PtrUnion.pu64); break;
1728 default:
1729 Bs3TestPrintf("op%u: imm%u: %.*Rhxs\n", iOperand, pThis->aOperands[iOperand].cbOp * 8,
1730 pThis->aOperands[iOperand].cbOp, *PtrUnion.pu64);
1731 break;
1732 }
1733 break;
1734
1735 case BS3CG1OPLOC_MEM:
1736 case BS3CG1OPLOC_MEM_RW:
1737 PtrUnion.pb = &pThis->pbDataPg[X86_PAGE_SIZE - pThis->aOperands[iOperand].off];
1738 switch (pThis->aOperands[iOperand].cbOp)
1739 {
1740 case 1: Bs3TestPrintf("op%u: mem08: %#04RX8\n", iOperand, *PtrUnion.pu8); break;
1741 case 2: Bs3TestPrintf("op%u: mem16: %#06RX16\n", iOperand, *PtrUnion.pu16); break;
1742 case 4: Bs3TestPrintf("op%u: mem32: %#010RX32\n", iOperand, *PtrUnion.pu32); break;
1743 case 8: Bs3TestPrintf("op%u: mem64: %#018RX64\n", iOperand, *PtrUnion.pu64); break;
1744 default:
1745 Bs3TestPrintf("op%u: mem%u: %.*Rhxs\n", iOperand, pThis->aOperands[iOperand].cbOp * 8,
1746 pThis->aOperands[iOperand].cbOp, *PtrUnion.pu64);
1747 break;
1748 }
1749 if (pThis->aOperands[iOperand].enmLocation == BS3CG1OPLOC_MEM_RW)
1750 {
1751 PtrUnion.pb = pThis->MemOp.ab;
1752 switch (pThis->aOperands[iOperand].cbOp)
1753 {
1754 case 1: Bs3TestPrintf("op%u: expect mem08: %#04RX8\n", iOperand, *PtrUnion.pu8); break;
1755 case 2: Bs3TestPrintf("op%u: expect mem16: %#06RX16\n", iOperand, *PtrUnion.pu16); break;
1756 case 4: Bs3TestPrintf("op%u: expect mem32: %#010RX32\n", iOperand, *PtrUnion.pu32); break;
1757 case 8: Bs3TestPrintf("op%u: expect mem64: %#018RX64\n", iOperand, *PtrUnion.pu64); break;
1758 default:
1759 Bs3TestPrintf("op%u: expect mem%u: %.*Rhxs\n", iOperand, pThis->aOperands[iOperand].cbOp * 8,
1760 pThis->aOperands[iOperand].cbOp, *PtrUnion.pu64);
1761 break;
1762 }
1763 }
1764 break;
1765 }
1766 }
1767
1768 /* Display contexts: */
1769 Bs3TestPrintf("-- Expected context:\n");
1770 Bs3RegCtxPrint(&pThis->Ctx);
1771 Bs3TestPrintf("-- Actual context:\n");
1772 Bs3TrapPrintFrame(&pThis->TrapFrame);
1773 Bs3TestPrintf("\n");
1774 return false;
1775}
1776
1777
1778/**
1779 * Destroys the state, freeing all allocations and such.
1780 *
1781 * @param pThis The state.
1782 */
1783static void Bs3Cg1Destroy(PBS3CG1STATE pThis)
1784{
1785 if (BS3_MODE_IS_PAGED(pThis->bMode))
1786 {
1787#if ARCH_BITS != 16
1788 Bs3MemGuardedTestPageFree(pThis->pbCodePg);
1789 Bs3MemGuardedTestPageFree(pThis->pbDataPg);
1790#endif
1791 }
1792 else
1793 {
1794 Bs3MemFree(pThis->pbCodePg, X86_PAGE_SIZE);
1795 Bs3MemFree(pThis->pbDataPg, X86_PAGE_SIZE);
1796 }
1797}
1798
1799
1800/**
1801 * Initializes the state.
1802 *
1803 * @returns Success indicator (true/false)
1804 * @param pThis The state.
1805 * @param bMode The mode being tested.
1806 */
1807bool BS3_CMN_NM(Bs3Cg1Init)(PBS3CG1STATE pThis, uint8_t bMode, uint8_t cRings, uint8_t iFirstRing)
1808{
1809 BS3MEMKIND const enmMemKind = BS3_MODE_IS_RM_OR_V86(bMode) ? BS3MEMKIND_REAL
1810 : !BS3_MODE_IS_64BIT_CODE(bMode) ? BS3MEMKIND_TILED : BS3MEMKIND_FLAT32;
1811 unsigned iRing;
1812
1813 Bs3MemSet(pThis, 0, sizeof(*pThis));
1814
1815 pThis->bMode = bMode;
1816 pThis->pszMode = Bs3GetModeName(bMode);
1817 pThis->pszModeShort = Bs3GetModeNameShortLower(bMode);
1818 pThis->pchMnemonic = g_achBs3Cg1Mnemonics;
1819 pThis->pabOperands = g_abBs3Cg1Operands;
1820 pThis->pabOpcodes = g_abBs3Cg1Opcodes;
1821 pThis->fAdvanceMnemonic = 1;
1822
1823 /* Allocate guarded exectuable and data memory. */
1824 if (BS3_MODE_IS_PAGED(bMode))
1825 {
1826#if ARCH_BITS != 16
1827 pThis->pbCodePg = Bs3MemGuardedTestPageAlloc(enmMemKind);
1828 if (!pThis->pbCodePg)
1829 return Bs3TestFailedF("First Bs3MemGuardedTestPageAlloc(%d) failed", enmMemKind);
1830 pThis->pbDataPg = Bs3MemGuardedTestPageAlloc(enmMemKind);
1831 if (!pThis->pbDataPg)
1832 {
1833 Bs3MemGuardedTestPageFree(pThis->pbCodePg);
1834 return Bs3TestFailedF("Second Bs3MemGuardedTestPageAlloc(%d) failed", enmMemKind);
1835 }
1836 if ( BS3_MODE_IS_64BIT_CODE(bMode)
1837 && (uintptr_t)pThis->pbDataPg >= _2G)
1838 {
1839 Bs3TestFailedF("pbDataPg=%p is above 2GB and not simple to address from 64-bit code", pThis->pbDataPg);
1840 Bs3MemGuardedTestPageFree(pThis->pbDataPg);
1841 Bs3MemGuardedTestPageFree(pThis->pbCodePg);
1842 return 0;
1843 }
1844#else
1845 return Bs3TestFailed("WTF?! #1");
1846#endif
1847 }
1848 else
1849 {
1850 pThis->pbCodePg = Bs3MemAlloc(enmMemKind, X86_PAGE_SIZE);
1851 if (!pThis->pbCodePg)
1852 return Bs3TestFailedF("First Bs3MemAlloc(%d,Pg) failed", enmMemKind);
1853 pThis->pbDataPg = Bs3MemAlloc(enmMemKind, X86_PAGE_SIZE);
1854 if (!pThis->pbDataPg)
1855 {
1856 Bs3MemFree(pThis->pbCodePg, X86_PAGE_SIZE);
1857 return Bs3TestFailedF("Second Bs3MemAlloc(%d,Pg) failed", enmMemKind);
1858 }
1859 }
1860 pThis->uCodePgFlat = Bs3SelPtrToFlat(pThis->pbCodePg);
1861 pThis->uDataPgFlat = Bs3SelPtrToFlat(pThis->pbDataPg);
1862#if ARCH_BITS == 16
1863 pThis->CodePgFar.sel = BS3_FP_SEG(pThis->pbCodePg);
1864 pThis->CodePgFar.off = BS3_FP_OFF(pThis->pbCodePg);
1865 pThis->CodePgRip = BS3_FP_OFF(pThis->pbCodePg);
1866 pThis->DataPgFar.sel = BS3_FP_SEG(pThis->pbDataPg);
1867 pThis->DataPgFar.off = BS3_FP_OFF(pThis->pbDataPg);
1868#else
1869 if (BS3_MODE_IS_RM_OR_V86(bMode))
1870 {
1871 *(uint32_t *)&pThis->DataPgFar = Bs3SelFlatDataToRealMode(pThis->uDataPgFlat);
1872 ASMCompilerBarrier();
1873 pThis->CodePgFar.off = 0;
1874 pThis->CodePgFar.sel = pThis->uDataPgFlat >> 4;
1875 pThis->CodePgRip = pThis->CodePgFar.off;
1876 }
1877 else if (BS3_MODE_IS_16BIT_CODE(bMode))
1878 {
1879 *(uint32_t *)&pThis->DataPgFar = Bs3SelFlatDataToProtFar16(pThis->uDataPgFlat);
1880 ASMCompilerBarrier();
1881 pThis->CodePgFar.sel = BS3_SEL_SPARE_00;
1882 pThis->CodePgFar.off = 0;
1883 pThis->CodePgRip = 0;
1884 }
1885 else if (BS3_MODE_IS_32BIT_CODE(bMode))
1886 {
1887 *(uint32_t *)&pThis->DataPgFar = Bs3SelFlatDataToProtFar16(pThis->uDataPgFlat);
1888 ASMCompilerBarrier();
1889 pThis->CodePgFar.sel = 0;
1890 pThis->CodePgFar.off = 0;
1891 pThis->CodePgRip = (uintptr_t)pThis->pbCodePg;
1892 }
1893 else
1894 {
1895 pThis->DataPgFar.off = 0;
1896 pThis->DataPgFar.sel = 0;
1897 pThis->CodePgFar.off = 0;
1898 pThis->CodePgFar.sel = 0;
1899 pThis->CodePgRip = (uintptr_t)pThis->pbCodePg;
1900 }
1901#endif
1902
1903 /* Create basic context for each target ring. In protected 16-bit code we need
1904 set up code selectors that can access pbCodePg. ASSUMES 16-bit driver code! */
1905 Bs3RegCtxSaveEx(&pThis->aInitialCtxs[iFirstRing], bMode, 512);
1906 if (BS3_MODE_IS_RM_OR_V86(bMode))
1907 {
1908 pThis->aInitialCtxs[iFirstRing].cs = pThis->CodePgFar.sel;
1909 BS3_ASSERT(iFirstRing + 1 >= cRings);
1910 }
1911 else if (BS3_MODE_IS_16BIT_CODE(bMode))
1912 {
1913#if ARCH_BITS == 16
1914 uintptr_t const uFlatCodePgSeg = Bs3SelPtrToFlat(BS3_FP_MAKE(BS3_FP_SEG(pThis->pbCodePg), 0));
1915#else
1916 uintptr_t const uFlatCodePgSeg = (uintptr_t)pThis->pbCodePg;
1917#endif
1918 for (iRing = iFirstRing + 1; iRing < cRings; iRing++)
1919 {
1920 Bs3MemCpy(&pThis->aInitialCtxs[iRing], &pThis->aInitialCtxs[iFirstRing], sizeof(pThis->aInitialCtxs[iRing]));
1921 Bs3RegCtxConvertToRingX(&pThis->aInitialCtxs[iRing], iRing);
1922 }
1923 for (iRing = iFirstRing; iRing < cRings; iRing++)
1924 {
1925 pThis->aInitialCtxs[iRing].cs = BS3_SEL_SPARE_00 + iRing * 8 + iRing;
1926 Bs3SelSetup16BitCode(&Bs3GdteSpare00 + iRing, uFlatCodePgSeg, iRing);
1927 }
1928 }
1929 else
1930 {
1931 Bs3RegCtxSetRipCsFromCurPtr(&pThis->aInitialCtxs[iFirstRing], (FPFNBS3FAR)pThis->pbCodePg);
1932 for (iRing = iFirstRing + 1; iRing < cRings; iRing++)
1933 {
1934 Bs3MemCpy(&pThis->aInitialCtxs[iRing], &pThis->aInitialCtxs[iFirstRing], sizeof(pThis->aInitialCtxs[iRing]));
1935 Bs3RegCtxConvertToRingX(&pThis->aInitialCtxs[iRing], iRing);
1936 }
1937 }
1938
1939 ASMCompilerBarrier();
1940 return true;
1941}
1942
1943
1944static uint8_t BS3_CMN_NM(Bs3Cg1WorkerInner)(PBS3CG1STATE pThis, uint8_t bMode, uint8_t const cRings, uint8_t const iFirstRing)
1945{
1946 uint8_t iRing;
1947 unsigned iInstr;
1948
1949#if 0
1950 if (bMode != BS3_MODE_LM16)
1951 return BS3TESTDOMODE_SKIPPED;
1952#endif
1953
1954 /*
1955 * Test the instructions.
1956 */
1957 for (iInstr = 0; iInstr < g_cBs3Cg1Instructions;
1958 iInstr++,
1959 pThis->pchMnemonic += pThis->fAdvanceMnemonic * pThis->cchMnemonic,
1960 pThis->pabOperands += pThis->cOperands,
1961 pThis->pabOpcodes += pThis->cbOpcodes)
1962 {
1963 unsigned iEncoding;
1964 unsigned iEncodingNext;
1965 bool fInvalidInstr = false;
1966 uint8_t bTestXcptExpected = BS3_MODE_IS_PAGED(bMode) ? X86_XCPT_PF : X86_XCPT_UD;
1967
1968 /*
1969 * Expand the instruction information into the state.
1970 * Note! 16-bit will switch to a two level test header lookup once we exceed 64KB.
1971 */
1972 PCBS3CG1INSTR pInstr = &g_aBs3Cg1Instructions[iInstr];
1973 pThis->iInstr = iInstr;
1974 pThis->pTestHdr = (PCBS3CG1TESTHDR)&g_abBs3Cg1Tests[pInstr->offTests];
1975 pThis->fFlags = pInstr->fFlags;
1976 pThis->enmEncoding = (BS3CG1ENC)pInstr->enmEncoding;
1977 pThis->enmCpuTest = (BS3CG1CPU)pInstr->enmCpuTest;
1978 pThis->cchMnemonic = pInstr->cchMnemonic;
1979 if (pThis->fAdvanceMnemonic)
1980 Bs3TestSubF("%s / %.*s", pThis->pszModeShort, pThis->cchMnemonic, pThis->pchMnemonic);
1981 pThis->fAdvanceMnemonic = pInstr->fAdvanceMnemonic;
1982 pThis->cOperands = pInstr->cOperands;
1983 pThis->cbOpcodes = pInstr->cbOpcodes;
1984 switch (pThis->cOperands)
1985 {
1986 case 3: pThis->aenmOperands[3] = (BS3CG1OP)pThis->pabOperands[3];
1987 case 2: pThis->aenmOperands[2] = (BS3CG1OP)pThis->pabOperands[2];
1988 case 1: pThis->aenmOperands[1] = (BS3CG1OP)pThis->pabOperands[1];
1989 case 0: pThis->aenmOperands[0] = (BS3CG1OP)pThis->pabOperands[0];
1990 }
1991
1992 switch (pThis->cbOpcodes)
1993 {
1994 case 3: pThis->abOpcodes[3] = pThis->pabOpcodes[3];
1995 case 2: pThis->abOpcodes[2] = pThis->pabOpcodes[2];
1996 case 1: pThis->abOpcodes[1] = pThis->pabOpcodes[1];
1997 case 0: pThis->abOpcodes[0] = pThis->pabOpcodes[0];
1998 }
1999
2000 /*
2001 * Check if the CPU supports the instruction.
2002 */
2003 if (!Bs3Cg1CpuTestAndEnable(pThis))
2004 {
2005 fInvalidInstr = true;
2006 bTestXcptExpected = X86_XCPT_UD;
2007 }
2008
2009 /*
2010 * Prep the operands and encoding handling.
2011 */
2012 if (!Bs3Cg1EncodePrep(pThis))
2013 continue;
2014
2015 /*
2016 * Encode the instruction in various ways and check out the test values.
2017 */
2018 for (iEncoding = 0;; iEncoding = iEncodingNext)
2019 {
2020 /*
2021 * Encode the next instruction variation.
2022 */
2023 iEncodingNext = Bs3Cg1EncodeNext(pThis, iEncoding);
2024 if (iEncodingNext <= iEncoding)
2025 break;
2026 BS3CG1_DPRINTF(("\ndbg: Encoding #%u: cbCurInst=%u: %.*Rhxs\n",
2027 iEncoding, pThis->cbCurInstr, pThis->cbCurInstr, pThis->abCurInstr));
2028
2029 /*
2030 * Do the rings.
2031 */
2032 for (iRing = iFirstRing + pThis->fSameRingNotOkay; iRing < cRings; iRing++)
2033 {
2034 PCBS3CG1TESTHDR pHdr;
2035
2036 pThis->uCpl = iRing;
2037 BS3CG1_DPRINTF(("dbg: Ring %u\n", iRing));
2038
2039 /*
2040 * Do the tests one by one.
2041 */
2042 pHdr = pThis->pTestHdr;
2043 for (pThis->iTest = 0;; pThis->iTest++)
2044 {
2045 if (Bs3Cg1RunSelector(pThis, pHdr))
2046 {
2047 /* Okay, set up the execution context. */
2048 unsigned offCode;
2049 uint8_t BS3_FAR *pbCode;
2050
2051 Bs3MemCpy(&pThis->Ctx, &pThis->aInitialCtxs[iRing], sizeof(pThis->Ctx));
2052 if (BS3_MODE_IS_PAGED(bMode))
2053 {
2054 offCode = X86_PAGE_SIZE - pThis->cbCurInstr;
2055 pbCode = &pThis->pbCodePg[offCode];
2056 }
2057 else
2058 {
2059 pbCode = pThis->pbCodePg;
2060 pbCode[pThis->cbCurInstr] = 0x0f; /* UD2 */
2061 pbCode[pThis->cbCurInstr + 1] = 0x0b;
2062 offCode = 0;
2063 }
2064 pThis->Ctx.rip.u = pThis->CodePgRip + offCode;
2065 Bs3MemCpy(pbCode, pThis->abCurInstr, pThis->cbCurInstr);
2066
2067 if (Bs3Cg1RunContextModifier(pThis, &pThis->Ctx, pHdr, pHdr->cbSelector, pHdr->cbInput, NULL, pbCode))
2068 {
2069 /* Run the instruction. */
2070 BS3CG1_DPRINTF(("dbg: Running test #%u\n", pThis->iTest));
2071 //Bs3RegCtxPrint(&pThis->Ctx);
2072 Bs3TrapSetJmpAndRestore(&pThis->Ctx, &pThis->TrapFrame);
2073 BS3CG1_DPRINTF(("dbg: bXcpt=%#x rip=%RX64 -> %RX64\n",
2074 pThis->TrapFrame.bXcpt, pThis->Ctx.rip.u, pThis->TrapFrame.Ctx.rip.u));
2075
2076 /*
2077 * Apply the output modification program to the context.
2078 */
2079 pThis->Ctx.rflags.u32 &= ~X86_EFL_RF;
2080 pThis->Ctx.rflags.u32 |= pThis->TrapFrame.Ctx.rflags.u32 & X86_EFL_RF;
2081 pThis->bValueXcpt = UINT8_MAX;
2082 if ( fInvalidInstr
2083 || Bs3Cg1RunContextModifier(pThis, &pThis->Ctx, pHdr,
2084 pHdr->cbSelector + pHdr->cbInput, pHdr->cbOutput,
2085 &pThis->TrapFrame.Ctx, NULL /*pbCode*/))
2086 {
2087 Bs3Cg1CheckResult(pThis, fInvalidInstr, bTestXcptExpected, iEncoding);
2088 }
2089 }
2090 }
2091 else
2092 BS3CG1_DPRINTF(("dbg: Skipping #%u\n", pThis->iTest));
2093
2094 /* advance */
2095 if (pHdr->fLast)
2096 {
2097 BS3CG1_DPRINTF(("dbg: Last\n\n"));
2098 break;
2099 }
2100 pHdr = (PCBS3CG1TESTHDR)((uint8_t BS3_FAR *)(pHdr + 1) + pHdr->cbInput + pHdr->cbOutput + pHdr->cbSelector);
2101 }
2102 }
2103 }
2104
2105 /*
2106 * Clean up (segment registers, etc).
2107 */
2108 Bs3Cg1EncodeCleanup(pThis);
2109 }
2110
2111#if 0
2112 if (bMode >= BS3_MODE_PE16_32)
2113 {
2114 Bs3TestTerm();
2115 Bs3Shutdown();
2116 }
2117#endif
2118
2119 return 0;
2120}
2121
2122
2123BS3_DECL_FAR(uint8_t) BS3_CMN_NM(Bs3Cg1Worker)(uint8_t bMode)
2124{
2125 unsigned const iFirstRing = BS3_MODE_IS_V86(bMode) ? 3 : 0;
2126 uint8_t const cRings = BS3_MODE_IS_RM_OR_V86(bMode) ? 1 : 4;
2127 uint8_t bRet = 1;
2128#if 1
2129 BS3CG1STATE This;
2130 if (BS3_CMN_NM(Bs3Cg1Init)(&This, bMode, cRings, iFirstRing))
2131 {
2132 bRet = BS3_CMN_NM(Bs3Cg1WorkerInner)(&This, bMode, iFirstRing, cRings);
2133
2134 Bs3Cg1Destroy(&This);
2135 Bs3TestSubDone();
2136 }
2137#else
2138 PBS3CG1STATE pThis = (PBS3CG1STATE)Bs3MemAlloc(BS3MEMKIND_REAL, sizeof(*pThis));
2139 if (pThis)
2140 {
2141
2142 if (BS3_CMN_NM(Bs3Cg1Init)(pThis, bMode, cRings, iFirstRing))
2143 {
2144 bRet = BS3_CMN_NM(Bs3Cg1WorkerInner)(pThis, bMode, iFirstRing, cRings);
2145
2146 Bs3Cg1Destroy(pThis);
2147 Bs3TestSubDone();
2148 }
2149 Bs3MemFree(pThis, sizeof(*pThis));
2150 }
2151#endif
2152 return bRet;
2153}
2154
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette