VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/bootsectors/bs3-cpu-instr-2-template.c@ 65847

Last change on this file since 65847 was 65507, checked in by vboxsync, 8 years ago

bs3-cpu-instr-2: Simple cmpxchg16b testcase.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 30.3 KB
Line 
1/* $Id: bs3-cpu-instr-2-template.c 65507 2017-01-29 17:32:33Z vboxsync $ */
2/** @file
3 * BS3Kit - bs3-cpu-instr-2, C code template.
4 */
5
6/*
7 * Copyright (C) 2007-2016 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
28/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#include <iprt/asm.h>
32#include <iprt/asm-amd64-x86.h>
33
34
35
36
37/*********************************************************************************************************************************
38* Structures and Typedefs *
39*********************************************************************************************************************************/
40#ifdef BS3_INSTANTIATING_CMN
41#endif
42
43
44/*********************************************************************************************************************************
45* External Symbols *
46*********************************************************************************************************************************/
47#ifdef BS3_INSTANTIATING_CMN
48extern FNBS3FAR BS3_CMN_NM(bs3CpuInstr2_mul_xBX_ud2);
49extern FNBS3FAR BS3_CMN_NM(bs3CpuInstr2_imul_xBX_ud2);
50extern FNBS3FAR BS3_CMN_NM(bs3CpuInstr2_imul_xCX_xBX_ud2);
51extern FNBS3FAR BS3_CMN_NM(bs3CpuInstr2_div_xBX_ud2);
52extern FNBS3FAR BS3_CMN_NM(bs3CpuInstr2_idiv_xBX_ud2);
53# if ARCH_BITS == 64
54extern FNBS3FAR BS3_CMN_NM(bs3CpuInstr2_cmpxchg16b_rdi_ud2);
55extern FNBS3FAR BS3_CMN_NM(bs3CpuInstr2_lock_cmpxchg16b_rdi_ud2);
56# endif
57#endif
58
59
60/*********************************************************************************************************************************
61* Global Variables *
62*********************************************************************************************************************************/
63#ifdef BS3_INSTANTIATING_CMN
64
65#endif /* BS3_INSTANTIATING_CMN - global */
66
67
68/*
69 * Common code.
70 * Common code.
71 * Common code.
72 */
73#ifdef BS3_INSTANTIATING_CMN
74
75BS3_DECL_FAR(uint8_t) BS3_CMN_NM(bs3CpuInstr2_mul)(uint8_t bMode)
76{
77#define MUL_CHECK_EFLAGS_ZERO (uint16_t)(X86_EFL_AF | X86_EFL_ZF)
78#define MUL_CHECK_EFLAGS (uint16_t)(X86_EFL_CF | X86_EFL_OF | X86_EFL_SF | X86_EFL_PF)
79
80 static const struct
81 {
82 RTCCUINTREG uInAX;
83 RTCCUINTREG uInBX;
84 RTCCUINTREG uOutDX;
85 RTCCUINTREG uOutAX;
86 uint16_t fFlags;
87 } s_aTests[] =
88 {
89 { 1, 1,
90 0, 1, 0 },
91 { 2, 2,
92 0, 4, 0 },
93 { RTCCUINTREG_MAX, RTCCUINTREG_MAX,
94 RTCCUINTREG_MAX-1, 1, X86_EFL_CF | X86_EFL_OF },
95 { RTCCINTREG_MAX, RTCCINTREG_MAX,
96 RTCCINTREG_MAX / 2, 1, X86_EFL_CF | X86_EFL_OF },
97 { 1, RTCCUINTREG_MAX,
98 0, RTCCUINTREG_MAX, X86_EFL_PF | X86_EFL_SF },
99 { 1, RTCCINTREG_MAX,
100 0, RTCCINTREG_MAX, X86_EFL_PF },
101 { 2, RTCCINTREG_MAX,
102 0, RTCCUINTREG_MAX - 1, X86_EFL_SF },
103 { (RTCCUINTREG)RTCCINTREG_MAX + 1, 2,
104 1, 0, X86_EFL_PF | X86_EFL_CF | X86_EFL_OF },
105 { (RTCCUINTREG)RTCCINTREG_MAX / 2 + 1, 3,
106 0, ((RTCCUINTREG)RTCCINTREG_MAX / 2 + 1) * 3, X86_EFL_PF | X86_EFL_SF },
107 };
108
109 BS3REGCTX Ctx;
110 BS3TRAPFRAME TrapFrame;
111 unsigned i, j, k;
112
113 /* Ensure the structures are allocated before we sample the stack pointer. */
114 Bs3MemSet(&Ctx, 0, sizeof(Ctx));
115 Bs3MemSet(&TrapFrame, 0, sizeof(TrapFrame));
116
117 /*
118 * Create test context.
119 */
120 Bs3RegCtxSaveEx(&Ctx, bMode, 512);
121 Bs3RegCtxSetRipCsFromCurPtr(&Ctx, BS3_CMN_NM(bs3CpuInstr2_mul_xBX_ud2));
122 for (k = 0; k < 2; k++)
123 {
124 Ctx.rflags.u16 |= MUL_CHECK_EFLAGS | MUL_CHECK_EFLAGS_ZERO;
125 for (j = 0; j < 2; j++)
126 {
127 for (i = 0; i < RT_ELEMENTS(s_aTests); i++)
128 {
129 if (k == 0)
130 {
131 Ctx.rax.RT_CONCAT(u,ARCH_BITS) = s_aTests[i].uInAX;
132 Ctx.rbx.RT_CONCAT(u,ARCH_BITS) = s_aTests[i].uInBX;
133 }
134 else
135 {
136 Ctx.rax.RT_CONCAT(u,ARCH_BITS) = s_aTests[i].uInBX;
137 Ctx.rbx.RT_CONCAT(u,ARCH_BITS) = s_aTests[i].uInAX;
138 }
139 Bs3TrapSetJmpAndRestore(&Ctx, &TrapFrame);
140 if (TrapFrame.bXcpt != X86_XCPT_UD)
141 Bs3TestFailedF("Expected #UD got %#x", TrapFrame.bXcpt);
142 else if ( TrapFrame.Ctx.rax.RT_CONCAT(u,ARCH_BITS) != s_aTests[i].uOutAX
143 || TrapFrame.Ctx.rdx.RT_CONCAT(u,ARCH_BITS) != s_aTests[i].uOutDX
144 || (TrapFrame.Ctx.rflags.u16 & (MUL_CHECK_EFLAGS | MUL_CHECK_EFLAGS_ZERO))
145 != (s_aTests[i].fFlags & MUL_CHECK_EFLAGS) )
146 {
147 Bs3TestFailedF("test #%i failed: input %#" RTCCUINTREG_XFMT " * %#" RTCCUINTREG_XFMT,
148 i, s_aTests[i].uInAX, s_aTests[i].uInBX);
149
150 if (TrapFrame.Ctx.rax.RT_CONCAT(u,ARCH_BITS) != s_aTests[i].uOutAX)
151 Bs3TestFailedF("Expected xAX = %#RX" RT_XSTR(ARCH_BITS) " got %#RX" RT_XSTR(ARCH_BITS),
152 s_aTests[i].uOutAX, TrapFrame.Ctx.rax.RT_CONCAT(u,ARCH_BITS));
153 if (TrapFrame.Ctx.rdx.RT_CONCAT(u,ARCH_BITS) != s_aTests[i].uOutDX)
154 Bs3TestFailedF("Expected xDX = %#RX" RT_XSTR(ARCH_BITS) " got %#RX" RT_XSTR(ARCH_BITS),
155 s_aTests[i].uOutDX, TrapFrame.Ctx.rdx.RT_CONCAT(u,ARCH_BITS));
156 if ( (TrapFrame.Ctx.rflags.u16 & (MUL_CHECK_EFLAGS | MUL_CHECK_EFLAGS_ZERO))
157 != (s_aTests[i].fFlags & MUL_CHECK_EFLAGS) )
158 Bs3TestFailedF("Expected EFLAGS = %#06RX16, got %#06RX16", s_aTests[i].fFlags & MUL_CHECK_EFLAGS,
159 TrapFrame.Ctx.rflags.u16 & (MUL_CHECK_EFLAGS | MUL_CHECK_EFLAGS_ZERO));
160 }
161 }
162 Ctx.rflags.u16 &= ~(MUL_CHECK_EFLAGS | MUL_CHECK_EFLAGS_ZERO);
163 }
164 }
165
166 return 0;
167}
168
169
170BS3_DECL_FAR(uint8_t) BS3_CMN_NM(bs3CpuInstr2_imul)(uint8_t bMode)
171{
172#define IMUL_CHECK_EFLAGS_ZERO (uint16_t)(X86_EFL_AF | X86_EFL_ZF)
173#define IMUL_CHECK_EFLAGS (uint16_t)(X86_EFL_CF | X86_EFL_OF | X86_EFL_SF | X86_EFL_PF)
174 static const struct
175 {
176 RTCCUINTREG uInAX;
177 RTCCUINTREG uInBX;
178 RTCCUINTREG uOutDX;
179 RTCCUINTREG uOutAX;
180 uint16_t fFlags;
181 } s_aTests[] =
182 {
183 /* two positive values. */
184 { 1, 1,
185 0, 1, 0 },
186 { 2, 2,
187 0, 4, 0 },
188 { RTCCINTREG_MAX, RTCCINTREG_MAX,
189 RTCCINTREG_MAX/2, 1, X86_EFL_CF | X86_EFL_OF },
190 { 1, RTCCINTREG_MAX,
191 0, RTCCINTREG_MAX, X86_EFL_PF },
192 { 2, RTCCINTREG_MAX,
193 0, RTCCUINTREG_MAX - 1U, X86_EFL_CF | X86_EFL_OF | X86_EFL_SF },
194 { 2, RTCCINTREG_MAX / 2,
195 0, RTCCINTREG_MAX - 1U, 0 },
196 { 2, (RTCCINTREG_MAX / 2 + 1),
197 0, (RTCCUINTREG)RTCCINTREG_MAX + 1U, X86_EFL_CF | X86_EFL_OF | X86_EFL_SF | X86_EFL_PF },
198 { 4, (RTCCINTREG_MAX / 2 + 1),
199 1, 0, X86_EFL_CF | X86_EFL_OF | X86_EFL_PF },
200
201 /* negative and positive */
202 { -4, 3,
203 -1, -12, X86_EFL_SF },
204 { 32, -127,
205 -1, -4064, X86_EFL_SF },
206 { RTCCINTREG_MIN, 1,
207 -1, RTCCINTREG_MIN, X86_EFL_SF | X86_EFL_PF },
208 { RTCCINTREG_MIN, 2,
209 -1, 0, X86_EFL_CF | X86_EFL_OF | X86_EFL_PF },
210 { RTCCINTREG_MIN, 3,
211 -2, RTCCINTREG_MIN, X86_EFL_CF | X86_EFL_OF | X86_EFL_SF | X86_EFL_PF },
212 { RTCCINTREG_MIN, 4,
213 -2, 0, X86_EFL_CF | X86_EFL_OF | X86_EFL_PF },
214 { RTCCINTREG_MIN, RTCCINTREG_MAX,
215 RTCCINTREG_MIN / 2, RTCCINTREG_MIN, X86_EFL_CF | X86_EFL_OF | X86_EFL_SF | X86_EFL_PF },
216 { RTCCINTREG_MIN, RTCCINTREG_MAX - 1,
217 RTCCINTREG_MIN / 2 + 1, 0, X86_EFL_CF | X86_EFL_OF | X86_EFL_PF },
218
219 /* two negative values. */
220 { -4, -63,
221 0, 252, X86_EFL_PF },
222 { RTCCINTREG_MIN, RTCCINTREG_MIN,
223 RTCCUINTREG_MAX / 4 + 1, 0, X86_EFL_CF | X86_EFL_OF | X86_EFL_PF },
224 { RTCCINTREG_MIN, RTCCINTREG_MIN + 1,
225 RTCCUINTREG_MAX / 4, RTCCINTREG_MIN, X86_EFL_CF | X86_EFL_OF | X86_EFL_SF | X86_EFL_PF},
226 { RTCCINTREG_MIN + 1, RTCCINTREG_MIN + 1,
227 RTCCUINTREG_MAX / 4, 1, X86_EFL_CF | X86_EFL_OF },
228
229 };
230
231 BS3REGCTX Ctx;
232 BS3TRAPFRAME TrapFrame;
233 unsigned i, j, k;
234
235 /* Ensure the structures are allocated before we sample the stack pointer. */
236 Bs3MemSet(&Ctx, 0, sizeof(Ctx));
237 Bs3MemSet(&TrapFrame, 0, sizeof(TrapFrame));
238
239 /*
240 * Create test context.
241 */
242 Bs3RegCtxSaveEx(&Ctx, bMode, 512);
243 Bs3RegCtxSetRipCsFromCurPtr(&Ctx, BS3_CMN_NM(bs3CpuInstr2_imul_xBX_ud2));
244
245 for (k = 0; k < 2; k++)
246 {
247 Ctx.rflags.u16 |= MUL_CHECK_EFLAGS | MUL_CHECK_EFLAGS_ZERO;
248 for (j = 0; j < 2; j++)
249 {
250 for (i = 0; i < RT_ELEMENTS(s_aTests); i++)
251 {
252 if (k == 0)
253 {
254 Ctx.rax.RT_CONCAT(u,ARCH_BITS) = s_aTests[i].uInAX;
255 Ctx.rbx.RT_CONCAT(u,ARCH_BITS) = s_aTests[i].uInBX;
256 }
257 else
258 {
259 Ctx.rax.RT_CONCAT(u,ARCH_BITS) = s_aTests[i].uInBX;
260 Ctx.rbx.RT_CONCAT(u,ARCH_BITS) = s_aTests[i].uInAX;
261 }
262 Bs3TrapSetJmpAndRestore(&Ctx, &TrapFrame);
263 if (TrapFrame.bXcpt != X86_XCPT_UD)
264 Bs3TestFailedF("Expected #UD got %#x", TrapFrame.bXcpt);
265 else if ( TrapFrame.Ctx.rax.RT_CONCAT(u,ARCH_BITS) != s_aTests[i].uOutAX
266 || TrapFrame.Ctx.rdx.RT_CONCAT(u,ARCH_BITS) != s_aTests[i].uOutDX
267 || (TrapFrame.Ctx.rflags.u16 & (IMUL_CHECK_EFLAGS | IMUL_CHECK_EFLAGS_ZERO))
268 != (s_aTests[i].fFlags & IMUL_CHECK_EFLAGS) )
269 {
270 Bs3TestFailedF("test #%i failed: input %#" RTCCUINTREG_XFMT " * %#" RTCCUINTREG_XFMT,
271 i, s_aTests[i].uInAX, s_aTests[i].uInBX);
272
273 if (TrapFrame.Ctx.rax.RT_CONCAT(u,ARCH_BITS) != s_aTests[i].uOutAX)
274 Bs3TestFailedF("Expected xAX = %#RX" RT_XSTR(ARCH_BITS) " got %#RX" RT_XSTR(ARCH_BITS),
275 s_aTests[i].uOutAX, TrapFrame.Ctx.rax.RT_CONCAT(u,ARCH_BITS));
276 if (TrapFrame.Ctx.rdx.RT_CONCAT(u,ARCH_BITS) != s_aTests[i].uOutDX)
277 Bs3TestFailedF("Expected xDX = %#RX" RT_XSTR(ARCH_BITS) " got %#RX" RT_XSTR(ARCH_BITS),
278 s_aTests[i].uOutDX, TrapFrame.Ctx.rdx.RT_CONCAT(u,ARCH_BITS));
279 if ( (TrapFrame.Ctx.rflags.u16 & (IMUL_CHECK_EFLAGS | IMUL_CHECK_EFLAGS_ZERO))
280 != (s_aTests[i].fFlags & IMUL_CHECK_EFLAGS) )
281 Bs3TestFailedF("Expected EFLAGS = %#06RX16, got %#06RX16", s_aTests[i].fFlags & IMUL_CHECK_EFLAGS,
282 TrapFrame.Ctx.rflags.u16 & (IMUL_CHECK_EFLAGS | IMUL_CHECK_EFLAGS_ZERO));
283 }
284 }
285 }
286 }
287
288 /*
289 * Repeat for the truncating two operand version.
290 */
291 Bs3RegCtxSetRipCsFromCurPtr(&Ctx, BS3_CMN_NM(bs3CpuInstr2_imul_xCX_xBX_ud2));
292
293 for (k = 0; k < 2; k++)
294 {
295 Ctx.rflags.u16 |= MUL_CHECK_EFLAGS | MUL_CHECK_EFLAGS_ZERO;
296 for (j = 0; j < 2; j++)
297 {
298 for (i = 0; i < RT_ELEMENTS(s_aTests); i++)
299 {
300 if (k == 0)
301 {
302 Ctx.rcx.RT_CONCAT(u,ARCH_BITS) = s_aTests[i].uInAX;
303 Ctx.rbx.RT_CONCAT(u,ARCH_BITS) = s_aTests[i].uInBX;
304 }
305 else
306 {
307 Ctx.rcx.RT_CONCAT(u,ARCH_BITS) = s_aTests[i].uInBX;
308 Ctx.rbx.RT_CONCAT(u,ARCH_BITS) = s_aTests[i].uInAX;
309 }
310 Bs3TrapSetJmpAndRestore(&Ctx, &TrapFrame);
311 if (TrapFrame.bXcpt != X86_XCPT_UD)
312 Bs3TestFailedF("Expected #UD got %#x", TrapFrame.bXcpt);
313 else if ( TrapFrame.Ctx.rcx.RT_CONCAT(u,ARCH_BITS) != s_aTests[i].uOutAX
314 || TrapFrame.Ctx.rdx.u != Ctx.rdx.u
315 || TrapFrame.Ctx.rbx.u != Ctx.rbx.u
316 || (TrapFrame.Ctx.rflags.u16 & (IMUL_CHECK_EFLAGS | IMUL_CHECK_EFLAGS_ZERO))
317 != (s_aTests[i].fFlags & IMUL_CHECK_EFLAGS) )
318 {
319 Bs3TestFailedF("test #%i failed: input %#" RTCCUINTREG_XFMT " * %#" RTCCUINTREG_XFMT,
320 i, s_aTests[i].uInAX, s_aTests[i].uInBX);
321
322 if (TrapFrame.Ctx.rcx.RT_CONCAT(u,ARCH_BITS) != s_aTests[i].uOutAX)
323 Bs3TestFailedF("Expected xAX = %#RX" RT_XSTR(ARCH_BITS) " got %#RX" RT_XSTR(ARCH_BITS),
324 s_aTests[i].uOutAX, TrapFrame.Ctx.rcx.RT_CONCAT(u,ARCH_BITS));
325 if ( (TrapFrame.Ctx.rflags.u16 & (IMUL_CHECK_EFLAGS | IMUL_CHECK_EFLAGS_ZERO))
326 != (s_aTests[i].fFlags & IMUL_CHECK_EFLAGS) )
327 Bs3TestFailedF("Expected EFLAGS = %#06RX16, got %#06RX16", s_aTests[i].fFlags & IMUL_CHECK_EFLAGS,
328 TrapFrame.Ctx.rflags.u16 & (IMUL_CHECK_EFLAGS | IMUL_CHECK_EFLAGS_ZERO));
329 }
330 }
331 }
332 }
333
334 return 0;
335}
336
337
338BS3_DECL_FAR(uint8_t) BS3_CMN_NM(bs3CpuInstr2_div)(uint8_t bMode)
339{
340#define DIV_CHECK_EFLAGS (uint16_t)(X86_EFL_CF | X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF)
341 static const struct
342 {
343 RTCCUINTREG uInDX;
344 RTCCUINTREG uInAX;
345 RTCCUINTREG uInBX;
346 RTCCUINTREG uOutAX;
347 RTCCUINTREG uOutDX;
348 uint8_t bXcpt;
349 } s_aTests[] =
350 {
351 { 0, 1, 1,
352 1, 0, X86_XCPT_UD },
353 { 0, 5, 2,
354 2, 1, X86_XCPT_UD },
355 { 0, 0, 0,
356 0, 0, X86_XCPT_DE },
357 { RTCCUINTREG_MAX, RTCCUINTREG_MAX, 0,
358 0, 0, X86_XCPT_DE },
359 { RTCCUINTREG_MAX, RTCCUINTREG_MAX, 1,
360 0, 0, X86_XCPT_DE },
361 { RTCCUINTREG_MAX, RTCCUINTREG_MAX, RTCCUINTREG_MAX,
362 0, 0, X86_XCPT_DE },
363 { RTCCUINTREG_MAX - 1, RTCCUINTREG_MAX, RTCCUINTREG_MAX,
364 RTCCUINTREG_MAX, RTCCUINTREG_MAX - 1, X86_XCPT_UD },
365 };
366
367 BS3REGCTX Ctx;
368 BS3TRAPFRAME TrapFrame;
369 unsigned i, j;
370
371 /* Ensure the structures are allocated before we sample the stack pointer. */
372 Bs3MemSet(&Ctx, 0, sizeof(Ctx));
373 Bs3MemSet(&TrapFrame, 0, sizeof(TrapFrame));
374
375 /*
376 * Create test context.
377 */
378 Bs3RegCtxSaveEx(&Ctx, bMode, 512);
379 Bs3RegCtxSetRipCsFromCurPtr(&Ctx, BS3_CMN_NM(bs3CpuInstr2_div_xBX_ud2));
380
381 /*
382 * Do the tests twice, first with all flags set, then once again with
383 * flags cleared. The flags are not touched by my intel skylake CPU.
384 */
385 Ctx.rflags.u16 |= DIV_CHECK_EFLAGS;
386 for (j = 0; j < 2; j++)
387 {
388 for (i = 0; i < RT_ELEMENTS(s_aTests); i++)
389 {
390 Ctx.rax.RT_CONCAT(u,ARCH_BITS) = s_aTests[i].uInAX;
391 Ctx.rdx.RT_CONCAT(u,ARCH_BITS) = s_aTests[i].uInDX;
392 Ctx.rbx.RT_CONCAT(u,ARCH_BITS) = s_aTests[i].uInBX;
393 Bs3TrapSetJmpAndRestore(&Ctx, &TrapFrame);
394
395 if ( TrapFrame.bXcpt != s_aTests[i].bXcpt
396 || ( s_aTests[i].bXcpt == X86_XCPT_UD
397 ? TrapFrame.Ctx.rax.RT_CONCAT(u,ARCH_BITS) != s_aTests[i].uOutAX
398 || TrapFrame.Ctx.rdx.RT_CONCAT(u,ARCH_BITS) != s_aTests[i].uOutDX
399 || (TrapFrame.Ctx.rflags.u16 & DIV_CHECK_EFLAGS) != (Ctx.rflags.u16 & DIV_CHECK_EFLAGS)
400 : TrapFrame.Ctx.rax.u != Ctx.rax.u
401 || TrapFrame.Ctx.rdx.u != Ctx.rdx.u
402 || (TrapFrame.Ctx.rflags.u16 & DIV_CHECK_EFLAGS) != (Ctx.rflags.u16 & DIV_CHECK_EFLAGS) ) )
403 {
404 Bs3TestFailedF("test #%i failed: input %#" RTCCUINTREG_XFMT ":%" RTCCUINTREG_XFMT " / %#" RTCCUINTREG_XFMT,
405 i, s_aTests[i].uInDX, s_aTests[i].uInAX, s_aTests[i].uInBX);
406 if (TrapFrame.bXcpt != s_aTests[i].bXcpt)
407 Bs3TestFailedF("Expected bXcpt = %#x, got %#x", s_aTests[i].bXcpt, TrapFrame.bXcpt);
408 if (s_aTests[i].bXcpt == X86_XCPT_UD)
409 {
410 if (TrapFrame.Ctx.rax.RT_CONCAT(u, ARCH_BITS) != s_aTests[i].uOutAX)
411 Bs3TestFailedF("Expected xAX = %#" RTCCUINTREG_XFMT ", got %#" RTCCUINTREG_XFMT,
412 s_aTests[i].uOutAX, TrapFrame.Ctx.rax.RT_CONCAT(u,ARCH_BITS));
413 if (TrapFrame.Ctx.rdx.RT_CONCAT(u,ARCH_BITS) != s_aTests[i].uOutDX)
414 Bs3TestFailedF("Expected xDX = %#" RTCCUINTREG_XFMT ", got %#" RTCCUINTREG_XFMT,
415 s_aTests[i].uOutDX, TrapFrame.Ctx.rdx.RT_CONCAT(u,ARCH_BITS));
416 if ((TrapFrame.Ctx.rflags.u16 & DIV_CHECK_EFLAGS) != (Ctx.rflags.u16 & DIV_CHECK_EFLAGS))
417 Bs3TestFailedF("Expected EFLAGS = %#06RX16, got %#06RX16",
418 Ctx.rflags.u16 & DIV_CHECK_EFLAGS, TrapFrame.Ctx.rflags.u16 & DIV_CHECK_EFLAGS);
419 }
420 }
421 }
422 Ctx.rflags.u16 &= ~DIV_CHECK_EFLAGS;
423 }
424
425 return 0;
426}
427
428
429
430BS3_DECL_FAR(uint8_t) BS3_CMN_NM(bs3CpuInstr2_idiv)(uint8_t bMode)
431{
432#define IDIV_CHECK_EFLAGS (uint16_t)(X86_EFL_CF | X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF)
433 static const struct
434 {
435 RTCCUINTREG uInDX;
436 RTCCUINTREG uInAX;
437 RTCCUINTREG uInBX;
438 RTCCUINTREG uOutAX;
439 RTCCUINTREG uOutDX;
440 uint8_t bXcpt;
441 } s_aTests[] =
442 {
443 { 0, 0, 0,
444 0, 0, X86_XCPT_DE },
445 { RTCCINTREG_MAX, RTCCINTREG_MAX, 0,
446 0, 0, X86_XCPT_DE },
447 /* two positive values. */
448 { 0, 1, 1,
449 1, 0, X86_XCPT_UD },
450 { 0, 5, 2,
451 2, 1, X86_XCPT_UD },
452 { RTCCINTREG_MAX / 2, RTCCUINTREG_MAX / 2, RTCCINTREG_MAX,
453 RTCCINTREG_MAX, RTCCINTREG_MAX - 1, X86_XCPT_UD },
454 { RTCCINTREG_MAX / 2, RTCCUINTREG_MAX / 2 + 1, RTCCINTREG_MAX,
455 RTCCINTREG_MAX, RTCCINTREG_MAX - 1, X86_XCPT_DE },
456 /* negative dividend, positive divisor. */
457 { -1, -7, 2,
458 -3, -1, X86_XCPT_UD },
459 { RTCCINTREG_MIN / 2 + 1, 0, RTCCINTREG_MAX,
460 RTCCINTREG_MIN + 2, RTCCINTREG_MIN + 2, X86_XCPT_UD },
461 { RTCCINTREG_MIN / 2, 0, RTCCINTREG_MAX,
462 0, 0, X86_XCPT_DE },
463 /* positive dividend, negative divisor. */
464 { 0, 7, -2,
465 -3, 1, X86_XCPT_UD },
466 { RTCCINTREG_MAX / 2 + 1, RTCCINTREG_MAX, RTCCINTREG_MIN,
467 RTCCINTREG_MIN, RTCCINTREG_MAX, X86_XCPT_UD },
468 { RTCCINTREG_MAX / 2 + 1, (RTCCUINTREG)RTCCINTREG_MAX+1, RTCCINTREG_MIN,
469 0, 0, X86_XCPT_DE },
470 /* negative dividend, negative divisor. */
471 { -1, -7, -2,
472 3, -1, X86_XCPT_UD },
473 { RTCCINTREG_MIN / 2, 1, RTCCINTREG_MIN,
474 RTCCINTREG_MAX, RTCCINTREG_MIN + 1, X86_XCPT_UD },
475 { RTCCINTREG_MIN / 2, 2, RTCCINTREG_MIN,
476 RTCCINTREG_MAX, RTCCINTREG_MIN + 2, X86_XCPT_UD },
477 { RTCCINTREG_MIN / 2, 0, RTCCINTREG_MIN,
478 0, 0, X86_XCPT_DE },
479 };
480
481 BS3REGCTX Ctx;
482 BS3TRAPFRAME TrapFrame;
483 unsigned i, j;
484
485 /* Ensure the structures are allocated before we sample the stack pointer. */
486 Bs3MemSet(&Ctx, 0, sizeof(Ctx));
487 Bs3MemSet(&TrapFrame, 0, sizeof(TrapFrame));
488
489 /*
490 * Create test context.
491 */
492 Bs3RegCtxSaveEx(&Ctx, bMode, 512);
493 Bs3RegCtxSetRipCsFromCurPtr(&Ctx, BS3_CMN_NM(bs3CpuInstr2_idiv_xBX_ud2));
494
495 /*
496 * Do the tests twice, first with all flags set, then once again with
497 * flags cleared. The flags are not touched by my intel skylake CPU.
498 */
499 Ctx.rflags.u16 |= IDIV_CHECK_EFLAGS;
500 for (j = 0; j < 2; j++)
501 {
502 for (i = 0; i < RT_ELEMENTS(s_aTests); i++)
503 {
504 Ctx.rax.RT_CONCAT(u,ARCH_BITS) = s_aTests[i].uInAX;
505 Ctx.rdx.RT_CONCAT(u,ARCH_BITS) = s_aTests[i].uInDX;
506 Ctx.rbx.RT_CONCAT(u,ARCH_BITS) = s_aTests[i].uInBX;
507 Bs3TrapSetJmpAndRestore(&Ctx, &TrapFrame);
508
509 if ( TrapFrame.bXcpt != s_aTests[i].bXcpt
510 || ( s_aTests[i].bXcpt == X86_XCPT_UD
511 ? TrapFrame.Ctx.rax.RT_CONCAT(u,ARCH_BITS) != s_aTests[i].uOutAX
512 || TrapFrame.Ctx.rdx.RT_CONCAT(u,ARCH_BITS) != s_aTests[i].uOutDX
513 || (TrapFrame.Ctx.rflags.u16 & IDIV_CHECK_EFLAGS) != (Ctx.rflags.u16 & IDIV_CHECK_EFLAGS)
514 : TrapFrame.Ctx.rax.u != Ctx.rax.u
515 || TrapFrame.Ctx.rdx.u != Ctx.rdx.u
516 || (TrapFrame.Ctx.rflags.u16 & IDIV_CHECK_EFLAGS) != (Ctx.rflags.u16 & IDIV_CHECK_EFLAGS) ) )
517 {
518 Bs3TestFailedF("test #%i failed: input %#" RTCCUINTREG_XFMT ":%" RTCCUINTREG_XFMT " / %#" RTCCUINTREG_XFMT,
519 i, s_aTests[i].uInDX, s_aTests[i].uInAX, s_aTests[i].uInBX);
520 if (TrapFrame.bXcpt != s_aTests[i].bXcpt)
521 Bs3TestFailedF("Expected bXcpt = %#x, got %#x", s_aTests[i].bXcpt, TrapFrame.bXcpt);
522 if (s_aTests[i].bXcpt == X86_XCPT_UD)
523 {
524 if (TrapFrame.Ctx.rax.RT_CONCAT(u, ARCH_BITS) != s_aTests[i].uOutAX)
525 Bs3TestFailedF("Expected xAX = %#" RTCCUINTREG_XFMT ", got %#" RTCCUINTREG_XFMT,
526 s_aTests[i].uOutAX, TrapFrame.Ctx.rax.RT_CONCAT(u,ARCH_BITS));
527 if (TrapFrame.Ctx.rdx.RT_CONCAT(u,ARCH_BITS) != s_aTests[i].uOutDX)
528 Bs3TestFailedF("Expected xDX = %#" RTCCUINTREG_XFMT ", got %#" RTCCUINTREG_XFMT,
529 s_aTests[i].uOutDX, TrapFrame.Ctx.rdx.RT_CONCAT(u,ARCH_BITS));
530 if ((TrapFrame.Ctx.rflags.u16 & IDIV_CHECK_EFLAGS) != (Ctx.rflags.u16 & IDIV_CHECK_EFLAGS))
531 Bs3TestFailedF("Expected EFLAGS = %#06RX16, got %#06RX16",
532 Ctx.rflags.u16 & IDIV_CHECK_EFLAGS, TrapFrame.Ctx.rflags.u16 & IDIV_CHECK_EFLAGS);
533 }
534 }
535 }
536 Ctx.rflags.u16 &= ~IDIV_CHECK_EFLAGS;
537 }
538
539 return 0;
540}
541
542
543# if ARCH_BITS == 64
544BS3_DECL_FAR(uint8_t) BS3_CMN_NM(bs3CpuInstr2_cmpxchg16b)(uint8_t bMode)
545{
546 BS3REGCTX Ctx;
547 BS3REGCTX ExpectCtx;
548 BS3TRAPFRAME TrapFrame;
549 RTUINT128U au128[3];
550 PRTUINT128U pau128 = RT_ALIGN_PT(&au128[0], sizeof(RTUINT128U), PRTUINT128U);
551 bool const fSupportCX16 = RT_BOOL(ASMCpuId_ECX(1) & X86_CPUID_FEATURE_ECX_CX16);
552 unsigned iLocked;
553 unsigned iFlags;
554 unsigned offBuf;
555 unsigned iMatch;
556
557 /* Ensure the structures are allocated before we sample the stack pointer. */
558 Bs3MemSet(&Ctx, 0, sizeof(Ctx));
559 Bs3MemSet(&ExpectCtx, 0, sizeof(ExpectCtx));
560 Bs3MemSet(&TrapFrame, 0, sizeof(TrapFrame));
561 Bs3MemSet(pau128, 0, sizeof(pau128[0]) * 2);
562
563 /*
564 * Create test context.
565 */
566 Bs3RegCtxSaveEx(&Ctx, bMode, 512);
567 if (!fSupportCX16)
568 Bs3TestPrintf("Note! CMPXCHG16B is not supported by the CPU!\n");
569
570 /*
571 * One loop with the normal variant and one with the locked one
572 */
573 g_usBs3TestStep = 0;
574 Bs3RegCtxSetRipCsFromCurPtr(&Ctx, BS3_CMN_NM(bs3CpuInstr2_cmpxchg16b_rdi_ud2));
575 for (iLocked = 0; iLocked < 2; iLocked++)
576 {
577 /*
578 * One loop with all status flags set, and one with them clear.
579 */
580 Ctx.rflags.u16 |= X86_EFL_STATUS_BITS;
581 for (iFlags = 0; iFlags < 2; iFlags++)
582 {
583 Bs3MemCpy(&ExpectCtx, &Ctx, sizeof(ExpectCtx));
584
585 for (offBuf = 0; offBuf < sizeof(RTUINT128U); offBuf++)
586 {
587# define CX16_OLD_LO UINT64_C(0xabb6345dcc9c4bbd)
588# define CX16_OLD_HI UINT64_C(0x7b06ea35749549ab)
589# define CX16_MISMATCH_LO UINT64_C(0xbace3e3590f18981)
590# define CX16_MISMATCH_HI UINT64_C(0x9b385e8bfd5b4000)
591# define CX16_STORE_LO UINT64_C(0x5cbd27d251f6559b)
592# define CX16_STORE_HI UINT64_C(0x17ff434ed1b54963)
593
594 PRTUINT128U pBuf = (PRTUINT128U)&pau128->au8[offBuf];
595
596 ExpectCtx.rax.u = Ctx.rax.u = CX16_MISMATCH_LO;
597 ExpectCtx.rdx.u = Ctx.rdx.u = CX16_MISMATCH_HI;
598 for (iMatch = 0; iMatch < 2; iMatch++)
599 {
600 uint8_t bExpectXcpt;
601 pBuf->s.Lo = CX16_OLD_LO;
602 pBuf->s.Hi = CX16_OLD_HI;
603 ExpectCtx.rdi.u = Ctx.rdi.u = (uintptr_t)pBuf;
604 Bs3TrapSetJmpAndRestore(&Ctx, &TrapFrame);
605 g_usBs3TestStep++;
606 //Bs3TestPrintf("Test: iFlags=%d offBuf=%d iMatch=%u\n", iFlags, offBuf, iMatch);
607 bExpectXcpt = X86_XCPT_UD;
608 if (fSupportCX16)
609 {
610 if (offBuf & 15)
611 {
612 bExpectXcpt = X86_XCPT_GP;
613 ExpectCtx.rip.u = Ctx.rip.u;
614 ExpectCtx.rflags.u32 = Ctx.rflags.u32;
615 }
616 else
617 {
618 ExpectCtx.rax.u = CX16_OLD_LO;
619 ExpectCtx.rdx.u = CX16_OLD_HI;
620 if (iMatch & 1)
621 ExpectCtx.rflags.u32 = Ctx.rflags.u32 | X86_EFL_ZF;
622 else
623 ExpectCtx.rflags.u32 = Ctx.rflags.u32 & ~X86_EFL_ZF;
624 ExpectCtx.rip.u = Ctx.rip.u + 4 + (iLocked & 1);
625 }
626 ExpectCtx.rflags.u32 |= X86_EFL_RF;
627 }
628 if ( !Bs3TestCheckRegCtxEx(&TrapFrame.Ctx, &ExpectCtx, 0 /*cbPcAdjust*/, 0 /*cbSpAdjust*/,
629 0 /*fExtraEfl*/, "lm64", 0 /*idTestStep*/)
630 || TrapFrame.bXcpt != bExpectXcpt)
631 {
632 if (TrapFrame.bXcpt != bExpectXcpt)
633 Bs3TestFailedF("Expected bXcpt=#%x, got %#x (%#x)", bExpectXcpt, TrapFrame.bXcpt, TrapFrame.uErrCd);
634 Bs3TestFailedF("^^^ iLocked=%d iFlags=%d offBuf=%d iMatch=%u\n", iLocked, iFlags, offBuf, iMatch);
635 ASMHalt();
636 }
637
638 ExpectCtx.rax.u = Ctx.rax.u = CX16_OLD_LO;
639 ExpectCtx.rdx.u = Ctx.rdx.u = CX16_OLD_HI;
640 }
641 }
642 Ctx.rflags.u16 &= ~X86_EFL_STATUS_BITS;
643 }
644 Bs3RegCtxSetRipCsFromCurPtr(&Ctx, BS3_CMN_NM(bs3CpuInstr2_lock_cmpxchg16b_rdi_ud2));
645 }
646
647 return 0;
648
649}
650# endif /* ARCH_BITS == 64 */
651
652
653#endif /* BS3_INSTANTIATING_CMN */
654
655
656
657/*
658 * Mode specific code.
659 * Mode specific code.
660 * Mode specific code.
661 */
662#ifdef BS3_INSTANTIATING_MODE
663
664
665#endif /* BS3_INSTANTIATING_MODE */
666
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