VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMAll/IEMAll.cpp@ 64573

Last change on this file since 64573 was 64545, checked in by vboxsync, 8 years ago

IEM: Added per-instruction statistics (not release).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 573.8 KB
Line 
1/* $Id: IEMAll.cpp 64545 2016-11-04 01:58:05Z vboxsync $ */
2/** @file
3 * IEM - Interpreted Execution Manager - All Contexts.
4 */
5
6/*
7 * Copyright (C) 2011-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
18
19/** @page pg_iem IEM - Interpreted Execution Manager
20 *
21 * The interpreted exeuction manager (IEM) is for executing short guest code
22 * sequences that are causing too many exits / virtualization traps. It will
23 * also be used to interpret single instructions, thus replacing the selective
24 * interpreters in EM and IOM.
25 *
26 * Design goals:
27 * - Relatively small footprint, although we favour speed and correctness
28 * over size.
29 * - Reasonably fast.
30 * - Correctly handle lock prefixed instructions.
31 * - Complete instruction set - eventually.
32 * - Refactorable into a recompiler, maybe.
33 * - Replace EMInterpret*.
34 *
35 * Using the existing disassembler has been considered, however this is thought
36 * to conflict with speed as the disassembler chews things a bit too much while
37 * leaving us with a somewhat complicated state to interpret afterwards.
38 *
39 *
40 * The current code is very much work in progress. You've been warned!
41 *
42 *
43 * @section sec_iem_fpu_instr FPU Instructions
44 *
45 * On x86 and AMD64 hosts, the FPU instructions are implemented by executing the
46 * same or equivalent instructions on the host FPU. To make life easy, we also
47 * let the FPU prioritize the unmasked exceptions for us. This however, only
48 * works reliably when CR0.NE is set, i.e. when using \#MF instead the IRQ 13
49 * for FPU exception delivery, because with CR0.NE=0 there is a window where we
50 * can trigger spurious FPU exceptions.
51 *
52 * The guest FPU state is not loaded into the host CPU and kept there till we
53 * leave IEM because the calling conventions have declared an all year open
54 * season on much of the FPU state. For instance an innocent looking call to
55 * memcpy might end up using a whole bunch of XMM or MM registers if the
56 * particular implementation finds it worthwhile.
57 *
58 *
59 * @section sec_iem_logging Logging
60 *
61 * The IEM code uses the \"IEM\" log group for the main logging. The different
62 * logging levels/flags are generally used for the following purposes:
63 * - Level 1 (Log) : Errors, exceptions, interrupts and such major events.
64 * - Flow (LogFlow): Basic enter/exit IEM state info.
65 * - Level 2 (Log2): ?
66 * - Level 3 (Log3): More detailed enter/exit IEM state info.
67 * - Level 4 (Log4): Decoding mnemonics w/ EIP.
68 * - Level 5 (Log5): Decoding details.
69 * - Level 6 (Log6): Enables/disables the lockstep comparison with REM.
70 * - Level 7 (Log7): iret++ execution logging.
71 * - Level 8 (Log8): Memory writes.
72 * - Level 9 (Log9): Memory reads.
73 *
74 */
75
76/** @def IEM_VERIFICATION_MODE_MINIMAL
77 * Use for pitting IEM against EM or something else in ring-0 or raw-mode
78 * context. */
79#if defined(DOXYGEN_RUNNING)
80# define IEM_VERIFICATION_MODE_MINIMAL
81#endif
82//#define IEM_LOG_MEMORY_WRITES
83#define IEM_IMPLEMENTS_TASKSWITCH
84
85/* Disabled warning C4505: 'iemRaisePageFaultJmp' : unreferenced local function has been removed */
86#ifdef _MSC_VER
87# pragma warning(disable:4505)
88#endif
89
90
91/*********************************************************************************************************************************
92* Header Files *
93*********************************************************************************************************************************/
94#define LOG_GROUP LOG_GROUP_IEM
95#define VMCPU_INCL_CPUM_GST_CTX
96#include <VBox/vmm/iem.h>
97#include <VBox/vmm/cpum.h>
98#include <VBox/vmm/pdm.h>
99#include <VBox/vmm/pgm.h>
100#include <VBox/vmm/iom.h>
101#include <VBox/vmm/em.h>
102#include <VBox/vmm/hm.h>
103#include <VBox/vmm/tm.h>
104#include <VBox/vmm/dbgf.h>
105#include <VBox/vmm/dbgftrace.h>
106#ifdef VBOX_WITH_RAW_MODE_NOT_R0
107# include <VBox/vmm/patm.h>
108# if defined(VBOX_WITH_CALL_RECORD) || defined(REM_MONITOR_CODE_PAGES)
109# include <VBox/vmm/csam.h>
110# endif
111#endif
112#include "IEMInternal.h"
113#ifdef IEM_VERIFICATION_MODE_FULL
114# include <VBox/vmm/rem.h>
115# include <VBox/vmm/mm.h>
116#endif
117#include <VBox/vmm/vm.h>
118#include <VBox/log.h>
119#include <VBox/err.h>
120#include <VBox/param.h>
121#include <VBox/dis.h>
122#include <VBox/disopcode.h>
123#include <iprt/assert.h>
124#include <iprt/string.h>
125#include <iprt/x86.h>
126
127
128/*********************************************************************************************************************************
129* Structures and Typedefs *
130*********************************************************************************************************************************/
131/** @typedef PFNIEMOP
132 * Pointer to an opcode decoder function.
133 */
134
135/** @def FNIEMOP_DEF
136 * Define an opcode decoder function.
137 *
138 * We're using macors for this so that adding and removing parameters as well as
139 * tweaking compiler specific attributes becomes easier. See FNIEMOP_CALL
140 *
141 * @param a_Name The function name.
142 */
143
144/** @typedef PFNIEMOPRM
145 * Pointer to an opcode decoder function with RM byte.
146 */
147
148/** @def FNIEMOPRM_DEF
149 * Define an opcode decoder function with RM byte.
150 *
151 * We're using macors for this so that adding and removing parameters as well as
152 * tweaking compiler specific attributes becomes easier. See FNIEMOP_CALL_1
153 *
154 * @param a_Name The function name.
155 */
156
157#if defined(__GNUC__) && defined(RT_ARCH_X86)
158typedef VBOXSTRICTRC (__attribute__((__fastcall__)) * PFNIEMOP)(PVMCPU pVCpu);
159typedef VBOXSTRICTRC (__attribute__((__fastcall__)) * PFNIEMOPRM)(PVMCPU pVCpu, uint8_t bRm);
160# define FNIEMOP_DEF(a_Name) \
161 IEM_STATIC VBOXSTRICTRC __attribute__((__fastcall__, __nothrow__)) a_Name(PVMCPU pVCpu)
162# define FNIEMOP_DEF_1(a_Name, a_Type0, a_Name0) \
163 IEM_STATIC VBOXSTRICTRC __attribute__((__fastcall__, __nothrow__)) a_Name(PVMCPU pVCpu, a_Type0 a_Name0)
164# define FNIEMOP_DEF_2(a_Name, a_Type0, a_Name0, a_Type1, a_Name1) \
165 IEM_STATIC VBOXSTRICTRC __attribute__((__fastcall__, __nothrow__)) a_Name(PVMCPU pVCpu, a_Type0 a_Name0, a_Type1 a_Name1)
166
167#elif defined(_MSC_VER) && defined(RT_ARCH_X86)
168typedef VBOXSTRICTRC (__fastcall * PFNIEMOP)(PVMCPU pVCpu);
169typedef VBOXSTRICTRC (__fastcall * PFNIEMOPRM)(PVMCPU pVCpu, uint8_t bRm);
170# define FNIEMOP_DEF(a_Name) \
171 IEM_STATIC /*__declspec(naked)*/ VBOXSTRICTRC __fastcall a_Name(PVMCPU pVCpu) RT_NO_THROW_DEF
172# define FNIEMOP_DEF_1(a_Name, a_Type0, a_Name0) \
173 IEM_STATIC /*__declspec(naked)*/ VBOXSTRICTRC __fastcall a_Name(PVMCPU pVCpu, a_Type0 a_Name0) RT_NO_THROW_DEF
174# define FNIEMOP_DEF_2(a_Name, a_Type0, a_Name0, a_Type1, a_Name1) \
175 IEM_STATIC /*__declspec(naked)*/ VBOXSTRICTRC __fastcall a_Name(PVMCPU pVCpu, a_Type0 a_Name0, a_Type1 a_Name1) RT_NO_THROW_DEF
176
177#elif defined(__GNUC__)
178typedef VBOXSTRICTRC (* PFNIEMOP)(PVMCPU pVCpu);
179typedef VBOXSTRICTRC (* PFNIEMOPRM)(PVMCPU pVCpu, uint8_t bRm);
180# define FNIEMOP_DEF(a_Name) \
181 IEM_STATIC VBOXSTRICTRC __attribute__((__nothrow__)) a_Name(PVMCPU pVCpu)
182# define FNIEMOP_DEF_1(a_Name, a_Type0, a_Name0) \
183 IEM_STATIC VBOXSTRICTRC __attribute__((__nothrow__)) a_Name(PVMCPU pVCpu, a_Type0 a_Name0)
184# define FNIEMOP_DEF_2(a_Name, a_Type0, a_Name0, a_Type1, a_Name1) \
185 IEM_STATIC VBOXSTRICTRC __attribute__((__nothrow__)) a_Name(PVMCPU pVCpu, a_Type0 a_Name0, a_Type1 a_Name1)
186
187#else
188typedef VBOXSTRICTRC (* PFNIEMOP)(PVMCPU pVCpu);
189typedef VBOXSTRICTRC (* PFNIEMOPRM)(PVMCPU pVCpu, uint8_t bRm);
190# define FNIEMOP_DEF(a_Name) \
191 IEM_STATIC VBOXSTRICTRC a_Name(PVMCPU pVCpu) RT_NO_THROW_DEF
192# define FNIEMOP_DEF_1(a_Name, a_Type0, a_Name0) \
193 IEM_STATIC VBOXSTRICTRC a_Name(PVMCPU pVCpu, a_Type0 a_Name0) RT_NO_THROW_DEF
194# define FNIEMOP_DEF_2(a_Name, a_Type0, a_Name0, a_Type1, a_Name1) \
195 IEM_STATIC VBOXSTRICTRC a_Name(PVMCPU pVCpu, a_Type0 a_Name0, a_Type1 a_Name1) RT_NO_THROW_DEF
196
197#endif
198#define FNIEMOPRM_DEF(a_Name) FNIEMOP_DEF_1(a_Name, uint8_t, bRm)
199
200
201/**
202 * Selector descriptor table entry as fetched by iemMemFetchSelDesc.
203 */
204typedef union IEMSELDESC
205{
206 /** The legacy view. */
207 X86DESC Legacy;
208 /** The long mode view. */
209 X86DESC64 Long;
210} IEMSELDESC;
211/** Pointer to a selector descriptor table entry. */
212typedef IEMSELDESC *PIEMSELDESC;
213
214
215/*********************************************************************************************************************************
216* Defined Constants And Macros *
217*********************************************************************************************************************************/
218/** @def IEM_WITH_SETJMP
219 * Enables alternative status code handling using setjmps.
220 *
221 * This adds a bit of expense via the setjmp() call since it saves all the
222 * non-volatile registers. However, it eliminates return code checks and allows
223 * for more optimal return value passing (return regs instead of stack buffer).
224 */
225#if defined(DOXYGEN_RUNNING) || defined(RT_OS_WINDOWS) || 1
226# define IEM_WITH_SETJMP
227#endif
228
229/** Temporary hack to disable the double execution. Will be removed in favor
230 * of a dedicated execution mode in EM. */
231//#define IEM_VERIFICATION_MODE_NO_REM
232
233/** Used to shut up GCC warnings about variables that 'may be used uninitialized'
234 * due to GCC lacking knowledge about the value range of a switch. */
235#define IEM_NOT_REACHED_DEFAULT_CASE_RET() default: AssertFailedReturn(VERR_IPE_NOT_REACHED_DEFAULT_CASE)
236
237/** Variant of IEM_NOT_REACHED_DEFAULT_CASE_RET that returns a custom value. */
238#define IEM_NOT_REACHED_DEFAULT_CASE_RET2(a_RetValue) default: AssertFailedReturn(a_RetValue)
239
240/**
241 * Returns IEM_RETURN_ASPECT_NOT_IMPLEMENTED, and in debug builds logs the
242 * occation.
243 */
244#ifdef LOG_ENABLED
245# define IEM_RETURN_ASPECT_NOT_IMPLEMENTED() \
246 do { \
247 /*Log*/ LogAlways(("%s: returning IEM_RETURN_ASPECT_NOT_IMPLEMENTED (line %d)\n", __FUNCTION__, __LINE__)); \
248 return VERR_IEM_ASPECT_NOT_IMPLEMENTED; \
249 } while (0)
250#else
251# define IEM_RETURN_ASPECT_NOT_IMPLEMENTED() \
252 return VERR_IEM_ASPECT_NOT_IMPLEMENTED
253#endif
254
255/**
256 * Returns IEM_RETURN_ASPECT_NOT_IMPLEMENTED, and in debug builds logs the
257 * occation using the supplied logger statement.
258 *
259 * @param a_LoggerArgs What to log on failure.
260 */
261#ifdef LOG_ENABLED
262# define IEM_RETURN_ASPECT_NOT_IMPLEMENTED_LOG(a_LoggerArgs) \
263 do { \
264 LogAlways((LOG_FN_FMT ": ", __PRETTY_FUNCTION__)); LogAlways(a_LoggerArgs); \
265 /*LogFunc(a_LoggerArgs);*/ \
266 return VERR_IEM_ASPECT_NOT_IMPLEMENTED; \
267 } while (0)
268#else
269# define IEM_RETURN_ASPECT_NOT_IMPLEMENTED_LOG(a_LoggerArgs) \
270 return VERR_IEM_ASPECT_NOT_IMPLEMENTED
271#endif
272
273/**
274 * Call an opcode decoder function.
275 *
276 * We're using macors for this so that adding and removing parameters can be
277 * done as we please. See FNIEMOP_DEF.
278 */
279#define FNIEMOP_CALL(a_pfn) (a_pfn)(pVCpu)
280
281/**
282 * Call a common opcode decoder function taking one extra argument.
283 *
284 * We're using macors for this so that adding and removing parameters can be
285 * done as we please. See FNIEMOP_DEF_1.
286 */
287#define FNIEMOP_CALL_1(a_pfn, a0) (a_pfn)(pVCpu, a0)
288
289/**
290 * Call a common opcode decoder function taking one extra argument.
291 *
292 * We're using macors for this so that adding and removing parameters can be
293 * done as we please. See FNIEMOP_DEF_1.
294 */
295#define FNIEMOP_CALL_2(a_pfn, a0, a1) (a_pfn)(pVCpu, a0, a1)
296
297/**
298 * Check if we're currently executing in real or virtual 8086 mode.
299 *
300 * @returns @c true if it is, @c false if not.
301 * @param a_pVCpu The IEM state of the current CPU.
302 */
303#define IEM_IS_REAL_OR_V86_MODE(a_pVCpu) (CPUMIsGuestInRealOrV86ModeEx(IEM_GET_CTX(a_pVCpu)))
304
305/**
306 * Check if we're currently executing in virtual 8086 mode.
307 *
308 * @returns @c true if it is, @c false if not.
309 * @param a_pVCpu The cross context virtual CPU structure of the calling thread.
310 */
311#define IEM_IS_V86_MODE(a_pVCpu) (CPUMIsGuestInV86ModeEx(IEM_GET_CTX(a_pVCpu)))
312
313/**
314 * Check if we're currently executing in long mode.
315 *
316 * @returns @c true if it is, @c false if not.
317 * @param a_pVCpu The cross context virtual CPU structure of the calling thread.
318 */
319#define IEM_IS_LONG_MODE(a_pVCpu) (CPUMIsGuestInLongModeEx(IEM_GET_CTX(a_pVCpu)))
320
321/**
322 * Check if we're currently executing in real mode.
323 *
324 * @returns @c true if it is, @c false if not.
325 * @param a_pVCpu The cross context virtual CPU structure of the calling thread.
326 */
327#define IEM_IS_REAL_MODE(a_pVCpu) (CPUMIsGuestInRealModeEx(IEM_GET_CTX(a_pVCpu)))
328
329/**
330 * Returns a (const) pointer to the CPUMFEATURES for the guest CPU.
331 * @returns PCCPUMFEATURES
332 * @param a_pVCpu The cross context virtual CPU structure of the calling thread.
333 */
334#define IEM_GET_GUEST_CPU_FEATURES(a_pVCpu) (&((a_pVCpu)->CTX_SUFF(pVM)->cpum.ro.GuestFeatures))
335
336/**
337 * Returns a (const) pointer to the CPUMFEATURES for the host CPU.
338 * @returns PCCPUMFEATURES
339 * @param a_pVCpu The cross context virtual CPU structure of the calling thread.
340 */
341#define IEM_GET_HOST_CPU_FEATURES(a_pVCpu) (&((a_pVCpu)->CTX_SUFF(pVM)->cpum.ro.HostFeatures))
342
343/**
344 * Evaluates to true if we're presenting an Intel CPU to the guest.
345 */
346#define IEM_IS_GUEST_CPU_INTEL(a_pVCpu) ( (a_pVCpu)->iem.s.enmCpuVendor == CPUMCPUVENDOR_INTEL )
347
348/**
349 * Evaluates to true if we're presenting an AMD CPU to the guest.
350 */
351#define IEM_IS_GUEST_CPU_AMD(a_pVCpu) ( (a_pVCpu)->iem.s.enmCpuVendor == CPUMCPUVENDOR_AMD )
352
353/**
354 * Check if the address is canonical.
355 */
356#define IEM_IS_CANONICAL(a_u64Addr) X86_IS_CANONICAL(a_u64Addr)
357
358/** @def IEM_USE_UNALIGNED_DATA_ACCESS
359 * Use unaligned accesses instead of elaborate byte assembly. */
360#if defined(RT_ARCH_AMD64) || defined(RT_ARCH_X86) || defined(DOXYGEN_RUNNING)
361# define IEM_USE_UNALIGNED_DATA_ACCESS
362#endif
363
364
365/*********************************************************************************************************************************
366* Global Variables *
367*********************************************************************************************************************************/
368extern const PFNIEMOP g_apfnOneByteMap[256]; /* not static since we need to forward declare it. */
369
370
371/** Function table for the ADD instruction. */
372IEM_STATIC const IEMOPBINSIZES g_iemAImpl_add =
373{
374 iemAImpl_add_u8, iemAImpl_add_u8_locked,
375 iemAImpl_add_u16, iemAImpl_add_u16_locked,
376 iemAImpl_add_u32, iemAImpl_add_u32_locked,
377 iemAImpl_add_u64, iemAImpl_add_u64_locked
378};
379
380/** Function table for the ADC instruction. */
381IEM_STATIC const IEMOPBINSIZES g_iemAImpl_adc =
382{
383 iemAImpl_adc_u8, iemAImpl_adc_u8_locked,
384 iemAImpl_adc_u16, iemAImpl_adc_u16_locked,
385 iemAImpl_adc_u32, iemAImpl_adc_u32_locked,
386 iemAImpl_adc_u64, iemAImpl_adc_u64_locked
387};
388
389/** Function table for the SUB instruction. */
390IEM_STATIC const IEMOPBINSIZES g_iemAImpl_sub =
391{
392 iemAImpl_sub_u8, iemAImpl_sub_u8_locked,
393 iemAImpl_sub_u16, iemAImpl_sub_u16_locked,
394 iemAImpl_sub_u32, iemAImpl_sub_u32_locked,
395 iemAImpl_sub_u64, iemAImpl_sub_u64_locked
396};
397
398/** Function table for the SBB instruction. */
399IEM_STATIC const IEMOPBINSIZES g_iemAImpl_sbb =
400{
401 iemAImpl_sbb_u8, iemAImpl_sbb_u8_locked,
402 iemAImpl_sbb_u16, iemAImpl_sbb_u16_locked,
403 iemAImpl_sbb_u32, iemAImpl_sbb_u32_locked,
404 iemAImpl_sbb_u64, iemAImpl_sbb_u64_locked
405};
406
407/** Function table for the OR instruction. */
408IEM_STATIC const IEMOPBINSIZES g_iemAImpl_or =
409{
410 iemAImpl_or_u8, iemAImpl_or_u8_locked,
411 iemAImpl_or_u16, iemAImpl_or_u16_locked,
412 iemAImpl_or_u32, iemAImpl_or_u32_locked,
413 iemAImpl_or_u64, iemAImpl_or_u64_locked
414};
415
416/** Function table for the XOR instruction. */
417IEM_STATIC const IEMOPBINSIZES g_iemAImpl_xor =
418{
419 iemAImpl_xor_u8, iemAImpl_xor_u8_locked,
420 iemAImpl_xor_u16, iemAImpl_xor_u16_locked,
421 iemAImpl_xor_u32, iemAImpl_xor_u32_locked,
422 iemAImpl_xor_u64, iemAImpl_xor_u64_locked
423};
424
425/** Function table for the AND instruction. */
426IEM_STATIC const IEMOPBINSIZES g_iemAImpl_and =
427{
428 iemAImpl_and_u8, iemAImpl_and_u8_locked,
429 iemAImpl_and_u16, iemAImpl_and_u16_locked,
430 iemAImpl_and_u32, iemAImpl_and_u32_locked,
431 iemAImpl_and_u64, iemAImpl_and_u64_locked
432};
433
434/** Function table for the CMP instruction.
435 * @remarks Making operand order ASSUMPTIONS.
436 */
437IEM_STATIC const IEMOPBINSIZES g_iemAImpl_cmp =
438{
439 iemAImpl_cmp_u8, NULL,
440 iemAImpl_cmp_u16, NULL,
441 iemAImpl_cmp_u32, NULL,
442 iemAImpl_cmp_u64, NULL
443};
444
445/** Function table for the TEST instruction.
446 * @remarks Making operand order ASSUMPTIONS.
447 */
448IEM_STATIC const IEMOPBINSIZES g_iemAImpl_test =
449{
450 iemAImpl_test_u8, NULL,
451 iemAImpl_test_u16, NULL,
452 iemAImpl_test_u32, NULL,
453 iemAImpl_test_u64, NULL
454};
455
456/** Function table for the BT instruction. */
457IEM_STATIC const IEMOPBINSIZES g_iemAImpl_bt =
458{
459 NULL, NULL,
460 iemAImpl_bt_u16, NULL,
461 iemAImpl_bt_u32, NULL,
462 iemAImpl_bt_u64, NULL
463};
464
465/** Function table for the BTC instruction. */
466IEM_STATIC const IEMOPBINSIZES g_iemAImpl_btc =
467{
468 NULL, NULL,
469 iemAImpl_btc_u16, iemAImpl_btc_u16_locked,
470 iemAImpl_btc_u32, iemAImpl_btc_u32_locked,
471 iemAImpl_btc_u64, iemAImpl_btc_u64_locked
472};
473
474/** Function table for the BTR instruction. */
475IEM_STATIC const IEMOPBINSIZES g_iemAImpl_btr =
476{
477 NULL, NULL,
478 iemAImpl_btr_u16, iemAImpl_btr_u16_locked,
479 iemAImpl_btr_u32, iemAImpl_btr_u32_locked,
480 iemAImpl_btr_u64, iemAImpl_btr_u64_locked
481};
482
483/** Function table for the BTS instruction. */
484IEM_STATIC const IEMOPBINSIZES g_iemAImpl_bts =
485{
486 NULL, NULL,
487 iemAImpl_bts_u16, iemAImpl_bts_u16_locked,
488 iemAImpl_bts_u32, iemAImpl_bts_u32_locked,
489 iemAImpl_bts_u64, iemAImpl_bts_u64_locked
490};
491
492/** Function table for the BSF instruction. */
493IEM_STATIC const IEMOPBINSIZES g_iemAImpl_bsf =
494{
495 NULL, NULL,
496 iemAImpl_bsf_u16, NULL,
497 iemAImpl_bsf_u32, NULL,
498 iemAImpl_bsf_u64, NULL
499};
500
501/** Function table for the BSR instruction. */
502IEM_STATIC const IEMOPBINSIZES g_iemAImpl_bsr =
503{
504 NULL, NULL,
505 iemAImpl_bsr_u16, NULL,
506 iemAImpl_bsr_u32, NULL,
507 iemAImpl_bsr_u64, NULL
508};
509
510/** Function table for the IMUL instruction. */
511IEM_STATIC const IEMOPBINSIZES g_iemAImpl_imul_two =
512{
513 NULL, NULL,
514 iemAImpl_imul_two_u16, NULL,
515 iemAImpl_imul_two_u32, NULL,
516 iemAImpl_imul_two_u64, NULL
517};
518
519/** Group 1 /r lookup table. */
520IEM_STATIC const PCIEMOPBINSIZES g_apIemImplGrp1[8] =
521{
522 &g_iemAImpl_add,
523 &g_iemAImpl_or,
524 &g_iemAImpl_adc,
525 &g_iemAImpl_sbb,
526 &g_iemAImpl_and,
527 &g_iemAImpl_sub,
528 &g_iemAImpl_xor,
529 &g_iemAImpl_cmp
530};
531
532/** Function table for the INC instruction. */
533IEM_STATIC const IEMOPUNARYSIZES g_iemAImpl_inc =
534{
535 iemAImpl_inc_u8, iemAImpl_inc_u8_locked,
536 iemAImpl_inc_u16, iemAImpl_inc_u16_locked,
537 iemAImpl_inc_u32, iemAImpl_inc_u32_locked,
538 iemAImpl_inc_u64, iemAImpl_inc_u64_locked
539};
540
541/** Function table for the DEC instruction. */
542IEM_STATIC const IEMOPUNARYSIZES g_iemAImpl_dec =
543{
544 iemAImpl_dec_u8, iemAImpl_dec_u8_locked,
545 iemAImpl_dec_u16, iemAImpl_dec_u16_locked,
546 iemAImpl_dec_u32, iemAImpl_dec_u32_locked,
547 iemAImpl_dec_u64, iemAImpl_dec_u64_locked
548};
549
550/** Function table for the NEG instruction. */
551IEM_STATIC const IEMOPUNARYSIZES g_iemAImpl_neg =
552{
553 iemAImpl_neg_u8, iemAImpl_neg_u8_locked,
554 iemAImpl_neg_u16, iemAImpl_neg_u16_locked,
555 iemAImpl_neg_u32, iemAImpl_neg_u32_locked,
556 iemAImpl_neg_u64, iemAImpl_neg_u64_locked
557};
558
559/** Function table for the NOT instruction. */
560IEM_STATIC const IEMOPUNARYSIZES g_iemAImpl_not =
561{
562 iemAImpl_not_u8, iemAImpl_not_u8_locked,
563 iemAImpl_not_u16, iemAImpl_not_u16_locked,
564 iemAImpl_not_u32, iemAImpl_not_u32_locked,
565 iemAImpl_not_u64, iemAImpl_not_u64_locked
566};
567
568
569/** Function table for the ROL instruction. */
570IEM_STATIC const IEMOPSHIFTSIZES g_iemAImpl_rol =
571{
572 iemAImpl_rol_u8,
573 iemAImpl_rol_u16,
574 iemAImpl_rol_u32,
575 iemAImpl_rol_u64
576};
577
578/** Function table for the ROR instruction. */
579IEM_STATIC const IEMOPSHIFTSIZES g_iemAImpl_ror =
580{
581 iemAImpl_ror_u8,
582 iemAImpl_ror_u16,
583 iemAImpl_ror_u32,
584 iemAImpl_ror_u64
585};
586
587/** Function table for the RCL instruction. */
588IEM_STATIC const IEMOPSHIFTSIZES g_iemAImpl_rcl =
589{
590 iemAImpl_rcl_u8,
591 iemAImpl_rcl_u16,
592 iemAImpl_rcl_u32,
593 iemAImpl_rcl_u64
594};
595
596/** Function table for the RCR instruction. */
597IEM_STATIC const IEMOPSHIFTSIZES g_iemAImpl_rcr =
598{
599 iemAImpl_rcr_u8,
600 iemAImpl_rcr_u16,
601 iemAImpl_rcr_u32,
602 iemAImpl_rcr_u64
603};
604
605/** Function table for the SHL instruction. */
606IEM_STATIC const IEMOPSHIFTSIZES g_iemAImpl_shl =
607{
608 iemAImpl_shl_u8,
609 iemAImpl_shl_u16,
610 iemAImpl_shl_u32,
611 iemAImpl_shl_u64
612};
613
614/** Function table for the SHR instruction. */
615IEM_STATIC const IEMOPSHIFTSIZES g_iemAImpl_shr =
616{
617 iemAImpl_shr_u8,
618 iemAImpl_shr_u16,
619 iemAImpl_shr_u32,
620 iemAImpl_shr_u64
621};
622
623/** Function table for the SAR instruction. */
624IEM_STATIC const IEMOPSHIFTSIZES g_iemAImpl_sar =
625{
626 iemAImpl_sar_u8,
627 iemAImpl_sar_u16,
628 iemAImpl_sar_u32,
629 iemAImpl_sar_u64
630};
631
632
633/** Function table for the MUL instruction. */
634IEM_STATIC const IEMOPMULDIVSIZES g_iemAImpl_mul =
635{
636 iemAImpl_mul_u8,
637 iemAImpl_mul_u16,
638 iemAImpl_mul_u32,
639 iemAImpl_mul_u64
640};
641
642/** Function table for the IMUL instruction working implicitly on rAX. */
643IEM_STATIC const IEMOPMULDIVSIZES g_iemAImpl_imul =
644{
645 iemAImpl_imul_u8,
646 iemAImpl_imul_u16,
647 iemAImpl_imul_u32,
648 iemAImpl_imul_u64
649};
650
651/** Function table for the DIV instruction. */
652IEM_STATIC const IEMOPMULDIVSIZES g_iemAImpl_div =
653{
654 iemAImpl_div_u8,
655 iemAImpl_div_u16,
656 iemAImpl_div_u32,
657 iemAImpl_div_u64
658};
659
660/** Function table for the MUL instruction. */
661IEM_STATIC const IEMOPMULDIVSIZES g_iemAImpl_idiv =
662{
663 iemAImpl_idiv_u8,
664 iemAImpl_idiv_u16,
665 iemAImpl_idiv_u32,
666 iemAImpl_idiv_u64
667};
668
669/** Function table for the SHLD instruction */
670IEM_STATIC const IEMOPSHIFTDBLSIZES g_iemAImpl_shld =
671{
672 iemAImpl_shld_u16,
673 iemAImpl_shld_u32,
674 iemAImpl_shld_u64,
675};
676
677/** Function table for the SHRD instruction */
678IEM_STATIC const IEMOPSHIFTDBLSIZES g_iemAImpl_shrd =
679{
680 iemAImpl_shrd_u16,
681 iemAImpl_shrd_u32,
682 iemAImpl_shrd_u64,
683};
684
685
686/** Function table for the PUNPCKLBW instruction */
687IEM_STATIC const IEMOPMEDIAF1L1 g_iemAImpl_punpcklbw = { iemAImpl_punpcklbw_u64, iemAImpl_punpcklbw_u128 };
688/** Function table for the PUNPCKLBD instruction */
689IEM_STATIC const IEMOPMEDIAF1L1 g_iemAImpl_punpcklwd = { iemAImpl_punpcklwd_u64, iemAImpl_punpcklwd_u128 };
690/** Function table for the PUNPCKLDQ instruction */
691IEM_STATIC const IEMOPMEDIAF1L1 g_iemAImpl_punpckldq = { iemAImpl_punpckldq_u64, iemAImpl_punpckldq_u128 };
692/** Function table for the PUNPCKLQDQ instruction */
693IEM_STATIC const IEMOPMEDIAF1L1 g_iemAImpl_punpcklqdq = { NULL, iemAImpl_punpcklqdq_u128 };
694
695/** Function table for the PUNPCKHBW instruction */
696IEM_STATIC const IEMOPMEDIAF1H1 g_iemAImpl_punpckhbw = { iemAImpl_punpckhbw_u64, iemAImpl_punpckhbw_u128 };
697/** Function table for the PUNPCKHBD instruction */
698IEM_STATIC const IEMOPMEDIAF1H1 g_iemAImpl_punpckhwd = { iemAImpl_punpckhwd_u64, iemAImpl_punpckhwd_u128 };
699/** Function table for the PUNPCKHDQ instruction */
700IEM_STATIC const IEMOPMEDIAF1H1 g_iemAImpl_punpckhdq = { iemAImpl_punpckhdq_u64, iemAImpl_punpckhdq_u128 };
701/** Function table for the PUNPCKHQDQ instruction */
702IEM_STATIC const IEMOPMEDIAF1H1 g_iemAImpl_punpckhqdq = { NULL, iemAImpl_punpckhqdq_u128 };
703
704/** Function table for the PXOR instruction */
705IEM_STATIC const IEMOPMEDIAF2 g_iemAImpl_pxor = { iemAImpl_pxor_u64, iemAImpl_pxor_u128 };
706/** Function table for the PCMPEQB instruction */
707IEM_STATIC const IEMOPMEDIAF2 g_iemAImpl_pcmpeqb = { iemAImpl_pcmpeqb_u64, iemAImpl_pcmpeqb_u128 };
708/** Function table for the PCMPEQW instruction */
709IEM_STATIC const IEMOPMEDIAF2 g_iemAImpl_pcmpeqw = { iemAImpl_pcmpeqw_u64, iemAImpl_pcmpeqw_u128 };
710/** Function table for the PCMPEQD instruction */
711IEM_STATIC const IEMOPMEDIAF2 g_iemAImpl_pcmpeqd = { iemAImpl_pcmpeqd_u64, iemAImpl_pcmpeqd_u128 };
712
713
714#if defined(IEM_VERIFICATION_MODE_MINIMAL) || defined(IEM_LOG_MEMORY_WRITES)
715/** What IEM just wrote. */
716uint8_t g_abIemWrote[256];
717/** How much IEM just wrote. */
718size_t g_cbIemWrote;
719#endif
720
721
722/*********************************************************************************************************************************
723* Internal Functions *
724*********************************************************************************************************************************/
725IEM_STATIC VBOXSTRICTRC iemRaiseTaskSwitchFaultWithErr(PVMCPU pVCpu, uint16_t uErr);
726IEM_STATIC VBOXSTRICTRC iemRaiseTaskSwitchFaultCurrentTSS(PVMCPU pVCpu);
727IEM_STATIC VBOXSTRICTRC iemRaiseTaskSwitchFault0(PVMCPU pVCpu);
728IEM_STATIC VBOXSTRICTRC iemRaiseTaskSwitchFaultBySelector(PVMCPU pVCpu, uint16_t uSel);
729/*IEM_STATIC VBOXSTRICTRC iemRaiseSelectorNotPresent(PVMCPU pVCpu, uint32_t iSegReg, uint32_t fAccess);*/
730IEM_STATIC VBOXSTRICTRC iemRaiseSelectorNotPresentBySelector(PVMCPU pVCpu, uint16_t uSel);
731IEM_STATIC VBOXSTRICTRC iemRaiseSelectorNotPresentWithErr(PVMCPU pVCpu, uint16_t uErr);
732IEM_STATIC VBOXSTRICTRC iemRaiseStackSelectorNotPresentBySelector(PVMCPU pVCpu, uint16_t uSel);
733IEM_STATIC VBOXSTRICTRC iemRaiseStackSelectorNotPresentWithErr(PVMCPU pVCpu, uint16_t uErr);
734IEM_STATIC VBOXSTRICTRC iemRaiseGeneralProtectionFault(PVMCPU pVCpu, uint16_t uErr);
735IEM_STATIC VBOXSTRICTRC iemRaiseGeneralProtectionFault0(PVMCPU pVCpu);
736IEM_STATIC VBOXSTRICTRC iemRaiseGeneralProtectionFaultBySelector(PVMCPU pVCpu, RTSEL uSel);
737IEM_STATIC VBOXSTRICTRC iemRaiseSelectorBounds(PVMCPU pVCpu, uint32_t iSegReg, uint32_t fAccess);
738IEM_STATIC VBOXSTRICTRC iemRaiseSelectorBoundsBySelector(PVMCPU pVCpu, RTSEL Sel);
739IEM_STATIC VBOXSTRICTRC iemRaiseSelectorInvalidAccess(PVMCPU pVCpu, uint32_t iSegReg, uint32_t fAccess);
740IEM_STATIC VBOXSTRICTRC iemRaisePageFault(PVMCPU pVCpu, RTGCPTR GCPtrWhere, uint32_t fAccess, int rc);
741IEM_STATIC VBOXSTRICTRC iemRaiseAlignmentCheckException(PVMCPU pVCpu);
742#ifdef IEM_WITH_SETJMP
743DECL_NO_INLINE(IEM_STATIC, DECL_NO_RETURN(void)) iemRaisePageFaultJmp(PVMCPU pVCpu, RTGCPTR GCPtrWhere, uint32_t fAccess, int rc);
744DECL_NO_INLINE(IEM_STATIC, DECL_NO_RETURN(void)) iemRaiseGeneralProtectionFault0Jmp(PVMCPU pVCpu);
745DECL_NO_INLINE(IEM_STATIC, DECL_NO_RETURN(void)) iemRaiseSelectorBoundsJmp(PVMCPU pVCpu, uint32_t iSegReg, uint32_t fAccess);
746DECL_NO_INLINE(IEM_STATIC, DECL_NO_RETURN(void)) iemRaiseSelectorBoundsBySelectorJmp(PVMCPU pVCpu, RTSEL Sel);
747DECL_NO_INLINE(IEM_STATIC, DECL_NO_RETURN(void)) iemRaiseSelectorInvalidAccessJmp(PVMCPU pVCpu, uint32_t iSegReg, uint32_t fAccess);
748#endif
749
750IEM_STATIC VBOXSTRICTRC iemMemMap(PVMCPU pVCpu, void **ppvMem, size_t cbMem, uint8_t iSegReg, RTGCPTR GCPtrMem, uint32_t fAccess);
751IEM_STATIC VBOXSTRICTRC iemMemCommitAndUnmap(PVMCPU pVCpu, void *pvMem, uint32_t fAccess);
752IEM_STATIC VBOXSTRICTRC iemMemFetchDataU32(PVMCPU pVCpu, uint32_t *pu32Dst, uint8_t iSegReg, RTGCPTR GCPtrMem);
753IEM_STATIC VBOXSTRICTRC iemMemFetchDataU64(PVMCPU pVCpu, uint64_t *pu64Dst, uint8_t iSegReg, RTGCPTR GCPtrMem);
754IEM_STATIC VBOXSTRICTRC iemMemFetchSysU8(PVMCPU pVCpu, uint32_t *pu32Dst, uint8_t iSegReg, RTGCPTR GCPtrMem);
755IEM_STATIC VBOXSTRICTRC iemMemFetchSysU16(PVMCPU pVCpu, uint32_t *pu32Dst, uint8_t iSegReg, RTGCPTR GCPtrMem);
756IEM_STATIC VBOXSTRICTRC iemMemFetchSysU32(PVMCPU pVCpu, uint32_t *pu32Dst, uint8_t iSegReg, RTGCPTR GCPtrMem);
757IEM_STATIC VBOXSTRICTRC iemMemFetchSysU64(PVMCPU pVCpu, uint64_t *pu64Dst, uint8_t iSegReg, RTGCPTR GCPtrMem);
758IEM_STATIC VBOXSTRICTRC iemMemFetchSelDescWithErr(PVMCPU pVCpu, PIEMSELDESC pDesc, uint16_t uSel, uint8_t uXcpt, uint16_t uErrorCode);
759IEM_STATIC VBOXSTRICTRC iemMemFetchSelDesc(PVMCPU pVCpu, PIEMSELDESC pDesc, uint16_t uSel, uint8_t uXcpt);
760IEM_STATIC VBOXSTRICTRC iemMemStackPushCommitSpecial(PVMCPU pVCpu, void *pvMem, uint64_t uNewRsp);
761IEM_STATIC VBOXSTRICTRC iemMemStackPushBeginSpecial(PVMCPU pVCpu, size_t cbMem, void **ppvMem, uint64_t *puNewRsp);
762IEM_STATIC VBOXSTRICTRC iemMemStackPushU32(PVMCPU pVCpu, uint32_t u32Value);
763IEM_STATIC VBOXSTRICTRC iemMemStackPushU16(PVMCPU pVCpu, uint16_t u16Value);
764IEM_STATIC VBOXSTRICTRC iemMemMarkSelDescAccessed(PVMCPU pVCpu, uint16_t uSel);
765IEM_STATIC uint16_t iemSRegFetchU16(PVMCPU pVCpu, uint8_t iSegReg);
766
767#if defined(IEM_VERIFICATION_MODE_FULL) && !defined(IEM_VERIFICATION_MODE_MINIMAL)
768IEM_STATIC PIEMVERIFYEVTREC iemVerifyAllocRecord(PVMCPU pVCpu);
769#endif
770IEM_STATIC VBOXSTRICTRC iemVerifyFakeIOPortRead(PVMCPU pVCpu, RTIOPORT Port, uint32_t *pu32Value, size_t cbValue);
771IEM_STATIC VBOXSTRICTRC iemVerifyFakeIOPortWrite(PVMCPU pVCpu, RTIOPORT Port, uint32_t u32Value, size_t cbValue);
772
773
774
775/**
776 * Sets the pass up status.
777 *
778 * @returns VINF_SUCCESS.
779 * @param pVCpu The cross context virtual CPU structure of the
780 * calling thread.
781 * @param rcPassUp The pass up status. Must be informational.
782 * VINF_SUCCESS is not allowed.
783 */
784IEM_STATIC int iemSetPassUpStatus(PVMCPU pVCpu, VBOXSTRICTRC rcPassUp)
785{
786 AssertRC(VBOXSTRICTRC_VAL(rcPassUp)); Assert(rcPassUp != VINF_SUCCESS);
787
788 int32_t const rcOldPassUp = pVCpu->iem.s.rcPassUp;
789 if (rcOldPassUp == VINF_SUCCESS)
790 pVCpu->iem.s.rcPassUp = VBOXSTRICTRC_VAL(rcPassUp);
791 /* If both are EM scheduling codes, use EM priority rules. */
792 else if ( rcOldPassUp >= VINF_EM_FIRST && rcOldPassUp <= VINF_EM_LAST
793 && rcPassUp >= VINF_EM_FIRST && rcPassUp <= VINF_EM_LAST)
794 {
795 if (rcPassUp < rcOldPassUp)
796 {
797 Log(("IEM: rcPassUp=%Rrc! rcOldPassUp=%Rrc\n", VBOXSTRICTRC_VAL(rcPassUp), rcOldPassUp));
798 pVCpu->iem.s.rcPassUp = VBOXSTRICTRC_VAL(rcPassUp);
799 }
800 else
801 Log(("IEM: rcPassUp=%Rrc rcOldPassUp=%Rrc!\n", VBOXSTRICTRC_VAL(rcPassUp), rcOldPassUp));
802 }
803 /* Override EM scheduling with specific status code. */
804 else if (rcOldPassUp >= VINF_EM_FIRST && rcOldPassUp <= VINF_EM_LAST)
805 {
806 Log(("IEM: rcPassUp=%Rrc! rcOldPassUp=%Rrc\n", VBOXSTRICTRC_VAL(rcPassUp), rcOldPassUp));
807 pVCpu->iem.s.rcPassUp = VBOXSTRICTRC_VAL(rcPassUp);
808 }
809 /* Don't override specific status code, first come first served. */
810 else
811 Log(("IEM: rcPassUp=%Rrc rcOldPassUp=%Rrc!\n", VBOXSTRICTRC_VAL(rcPassUp), rcOldPassUp));
812 return VINF_SUCCESS;
813}
814
815
816/**
817 * Calculates the CPU mode.
818 *
819 * This is mainly for updating IEMCPU::enmCpuMode.
820 *
821 * @returns CPU mode.
822 * @param pCtx The register context for the CPU.
823 */
824DECLINLINE(IEMMODE) iemCalcCpuMode(PCPUMCTX pCtx)
825{
826 if (CPUMIsGuestIn64BitCodeEx(pCtx))
827 return IEMMODE_64BIT;
828 if (pCtx->cs.Attr.n.u1DefBig) /** @todo check if this is correct... */
829 return IEMMODE_32BIT;
830 return IEMMODE_16BIT;
831}
832
833
834/**
835 * Initializes the execution state.
836 *
837 * @param pVCpu The cross context virtual CPU structure of the
838 * calling thread.
839 * @param fBypassHandlers Whether to bypass access handlers.
840 *
841 * @remarks Callers of this must call iemUninitExec() to undo potentially fatal
842 * side-effects in strict builds.
843 */
844DECLINLINE(void) iemInitExec(PVMCPU pVCpu, bool fBypassHandlers)
845{
846 PCPUMCTX const pCtx = IEM_GET_CTX(pVCpu);
847
848 Assert(!VMCPU_FF_IS_SET(pVCpu, VMCPU_FF_IEM));
849
850#if defined(VBOX_STRICT) && (defined(IEM_VERIFICATION_MODE_FULL) || !defined(VBOX_WITH_RAW_MODE_NOT_R0))
851 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->cs));
852 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->ss));
853 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->es));
854 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->ds));
855 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->fs));
856 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->gs));
857 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->ldtr));
858 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->tr));
859#endif
860
861#ifdef VBOX_WITH_RAW_MODE_NOT_R0
862 CPUMGuestLazyLoadHiddenCsAndSs(pVCpu);
863#endif
864 pVCpu->iem.s.uCpl = CPUMGetGuestCPL(pVCpu);
865 pVCpu->iem.s.enmCpuMode = iemCalcCpuMode(pCtx);
866#ifdef VBOX_STRICT
867 pVCpu->iem.s.enmDefAddrMode = (IEMMODE)0xc0fe;
868 pVCpu->iem.s.enmEffAddrMode = (IEMMODE)0xc0fe;
869 pVCpu->iem.s.enmDefOpSize = (IEMMODE)0xc0fe;
870 pVCpu->iem.s.enmEffOpSize = (IEMMODE)0xc0fe;
871 pVCpu->iem.s.fPrefixes = 0xfeedbeef;
872 pVCpu->iem.s.uRexReg = 127;
873 pVCpu->iem.s.uRexB = 127;
874 pVCpu->iem.s.uRexIndex = 127;
875 pVCpu->iem.s.iEffSeg = 127;
876 pVCpu->iem.s.uFpuOpcode = UINT16_MAX;
877# ifdef IEM_WITH_CODE_TLB
878 pVCpu->iem.s.offInstrNextByte = UINT16_MAX;
879 pVCpu->iem.s.pbInstrBuf = NULL;
880 pVCpu->iem.s.cbInstrBuf = UINT16_MAX;
881 pVCpu->iem.s.cbInstrBufTotal = UINT16_MAX;
882 pVCpu->iem.s.offCurInstrStart = INT16_MAX;
883 pVCpu->iem.s.uInstrBufPc = UINT64_C(0xc0ffc0ffcff0c0ff);
884# else
885 pVCpu->iem.s.offOpcode = 127;
886 pVCpu->iem.s.cbOpcode = 127;
887# endif
888#endif
889
890 pVCpu->iem.s.cActiveMappings = 0;
891 pVCpu->iem.s.iNextMapping = 0;
892 pVCpu->iem.s.rcPassUp = VINF_SUCCESS;
893 pVCpu->iem.s.fBypassHandlers = fBypassHandlers;
894#ifdef VBOX_WITH_RAW_MODE_NOT_R0
895 pVCpu->iem.s.fInPatchCode = pVCpu->iem.s.uCpl == 0
896 && pCtx->cs.u64Base == 0
897 && pCtx->cs.u32Limit == UINT32_MAX
898 && PATMIsPatchGCAddr(pVCpu->CTX_SUFF(pVM), pCtx->eip);
899 if (!pVCpu->iem.s.fInPatchCode)
900 CPUMRawLeave(pVCpu, VINF_SUCCESS);
901#endif
902
903#ifdef IEM_VERIFICATION_MODE_FULL
904 pVCpu->iem.s.fNoRemSavedByExec = pVCpu->iem.s.fNoRem;
905 pVCpu->iem.s.fNoRem = true;
906#endif
907}
908
909
910/**
911 * Counterpart to #iemInitExec that undoes evil strict-build stuff.
912 *
913 * @param pVCpu The cross context virtual CPU structure of the
914 * calling thread.
915 */
916DECLINLINE(void) iemUninitExec(PVMCPU pVCpu)
917{
918 /* Note! do not touch fInPatchCode here! (see iemUninitExecAndFiddleStatusAndMaybeReenter) */
919#ifdef IEM_VERIFICATION_MODE_FULL
920 pVCpu->iem.s.fNoRem = pVCpu->iem.s.fNoRemSavedByExec;
921#endif
922#ifdef VBOX_STRICT
923# ifdef IEM_WITH_CODE_TLB
924# else
925 pVCpu->iem.s.cbOpcode = 0;
926# endif
927#else
928 NOREF(pVCpu);
929#endif
930}
931
932
933/**
934 * Initializes the decoder state.
935 *
936 * iemReInitDecoder is mostly a copy of this function.
937 *
938 * @param pVCpu The cross context virtual CPU structure of the
939 * calling thread.
940 * @param fBypassHandlers Whether to bypass access handlers.
941 */
942DECLINLINE(void) iemInitDecoder(PVMCPU pVCpu, bool fBypassHandlers)
943{
944 PCPUMCTX const pCtx = IEM_GET_CTX(pVCpu);
945
946 Assert(!VMCPU_FF_IS_SET(pVCpu, VMCPU_FF_IEM));
947
948#if defined(VBOX_STRICT) && (defined(IEM_VERIFICATION_MODE_FULL) || !defined(VBOX_WITH_RAW_MODE_NOT_R0))
949 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->cs));
950 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->ss));
951 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->es));
952 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->ds));
953 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->fs));
954 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->gs));
955 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->ldtr));
956 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->tr));
957#endif
958
959#ifdef VBOX_WITH_RAW_MODE_NOT_R0
960 CPUMGuestLazyLoadHiddenCsAndSs(pVCpu);
961#endif
962 pVCpu->iem.s.uCpl = CPUMGetGuestCPL(pVCpu);
963#ifdef IEM_VERIFICATION_MODE_FULL
964 if (pVCpu->iem.s.uInjectCpl != UINT8_MAX)
965 pVCpu->iem.s.uCpl = pVCpu->iem.s.uInjectCpl;
966#endif
967 IEMMODE enmMode = iemCalcCpuMode(pCtx);
968 pVCpu->iem.s.enmCpuMode = enmMode;
969 pVCpu->iem.s.enmDefAddrMode = enmMode; /** @todo check if this is correct... */
970 pVCpu->iem.s.enmEffAddrMode = enmMode;
971 if (enmMode != IEMMODE_64BIT)
972 {
973 pVCpu->iem.s.enmDefOpSize = enmMode; /** @todo check if this is correct... */
974 pVCpu->iem.s.enmEffOpSize = enmMode;
975 }
976 else
977 {
978 pVCpu->iem.s.enmDefOpSize = IEMMODE_32BIT;
979 pVCpu->iem.s.enmEffOpSize = IEMMODE_32BIT;
980 }
981 pVCpu->iem.s.fPrefixes = 0;
982 pVCpu->iem.s.uRexReg = 0;
983 pVCpu->iem.s.uRexB = 0;
984 pVCpu->iem.s.uRexIndex = 0;
985 pVCpu->iem.s.iEffSeg = X86_SREG_DS;
986#ifdef IEM_WITH_CODE_TLB
987 pVCpu->iem.s.pbInstrBuf = NULL;
988 pVCpu->iem.s.offInstrNextByte = 0;
989 pVCpu->iem.s.offCurInstrStart = 0;
990# ifdef VBOX_STRICT
991 pVCpu->iem.s.cbInstrBuf = UINT16_MAX;
992 pVCpu->iem.s.cbInstrBufTotal = UINT16_MAX;
993 pVCpu->iem.s.uInstrBufPc = UINT64_C(0xc0ffc0ffcff0c0ff);
994# endif
995#else
996 pVCpu->iem.s.offOpcode = 0;
997 pVCpu->iem.s.cbOpcode = 0;
998#endif
999 pVCpu->iem.s.cActiveMappings = 0;
1000 pVCpu->iem.s.iNextMapping = 0;
1001 pVCpu->iem.s.rcPassUp = VINF_SUCCESS;
1002 pVCpu->iem.s.fBypassHandlers = fBypassHandlers;
1003#ifdef VBOX_WITH_RAW_MODE_NOT_R0
1004 pVCpu->iem.s.fInPatchCode = pVCpu->iem.s.uCpl == 0
1005 && pCtx->cs.u64Base == 0
1006 && pCtx->cs.u32Limit == UINT32_MAX
1007 && PATMIsPatchGCAddr(pVCpu->CTX_SUFF(pVM), pCtx->eip);
1008 if (!pVCpu->iem.s.fInPatchCode)
1009 CPUMRawLeave(pVCpu, VINF_SUCCESS);
1010#endif
1011
1012#ifdef DBGFTRACE_ENABLED
1013 switch (enmMode)
1014 {
1015 case IEMMODE_64BIT:
1016 RTTraceBufAddMsgF(pVCpu->CTX_SUFF(pVM)->CTX_SUFF(hTraceBuf), "I64/%u %08llx", pVCpu->iem.s.uCpl, pCtx->rip);
1017 break;
1018 case IEMMODE_32BIT:
1019 RTTraceBufAddMsgF(pVCpu->CTX_SUFF(pVM)->CTX_SUFF(hTraceBuf), "I32/%u %04x:%08x", pVCpu->iem.s.uCpl, pCtx->cs.Sel, pCtx->eip);
1020 break;
1021 case IEMMODE_16BIT:
1022 RTTraceBufAddMsgF(pVCpu->CTX_SUFF(pVM)->CTX_SUFF(hTraceBuf), "I16/%u %04x:%04x", pVCpu->iem.s.uCpl, pCtx->cs.Sel, pCtx->eip);
1023 break;
1024 }
1025#endif
1026}
1027
1028
1029/**
1030 * Reinitializes the decoder state 2nd+ loop of IEMExecLots.
1031 *
1032 * This is mostly a copy of iemInitDecoder.
1033 *
1034 * @param pVCpu The cross context virtual CPU structure of the calling EMT.
1035 */
1036DECLINLINE(void) iemReInitDecoder(PVMCPU pVCpu)
1037{
1038 PCPUMCTX const pCtx = IEM_GET_CTX(pVCpu);
1039
1040 Assert(!VMCPU_FF_IS_SET(pVCpu, VMCPU_FF_IEM));
1041
1042#if defined(VBOX_STRICT) && (defined(IEM_VERIFICATION_MODE_FULL) || !defined(VBOX_WITH_RAW_MODE_NOT_R0))
1043 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->cs));
1044 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->ss));
1045 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->es));
1046 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->ds));
1047 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->fs));
1048 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->gs));
1049 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->ldtr));
1050 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->tr));
1051#endif
1052
1053 pVCpu->iem.s.uCpl = CPUMGetGuestCPL(pVCpu); /** @todo this should be updated during execution! */
1054#ifdef IEM_VERIFICATION_MODE_FULL
1055 if (pVCpu->iem.s.uInjectCpl != UINT8_MAX)
1056 pVCpu->iem.s.uCpl = pVCpu->iem.s.uInjectCpl;
1057#endif
1058 IEMMODE enmMode = iemCalcCpuMode(pCtx);
1059 pVCpu->iem.s.enmCpuMode = enmMode; /** @todo this should be updated during execution! */
1060 pVCpu->iem.s.enmDefAddrMode = enmMode; /** @todo check if this is correct... */
1061 pVCpu->iem.s.enmEffAddrMode = enmMode;
1062 if (enmMode != IEMMODE_64BIT)
1063 {
1064 pVCpu->iem.s.enmDefOpSize = enmMode; /** @todo check if this is correct... */
1065 pVCpu->iem.s.enmEffOpSize = enmMode;
1066 }
1067 else
1068 {
1069 pVCpu->iem.s.enmDefOpSize = IEMMODE_32BIT;
1070 pVCpu->iem.s.enmEffOpSize = IEMMODE_32BIT;
1071 }
1072 pVCpu->iem.s.fPrefixes = 0;
1073 pVCpu->iem.s.uRexReg = 0;
1074 pVCpu->iem.s.uRexB = 0;
1075 pVCpu->iem.s.uRexIndex = 0;
1076 pVCpu->iem.s.iEffSeg = X86_SREG_DS;
1077#ifdef IEM_WITH_CODE_TLB
1078 if (pVCpu->iem.s.pbInstrBuf)
1079 {
1080 uint64_t off = (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT ? pCtx->rip : pCtx->eip + (uint32_t)pCtx->cs.u64Base)
1081 - pVCpu->iem.s.uInstrBufPc;
1082 if (off < pVCpu->iem.s.cbInstrBufTotal)
1083 {
1084 pVCpu->iem.s.offInstrNextByte = (uint32_t)off;
1085 pVCpu->iem.s.offCurInstrStart = (uint16_t)off;
1086 if ((uint16_t)off + 15 <= pVCpu->iem.s.cbInstrBufTotal)
1087 pVCpu->iem.s.cbInstrBuf = (uint16_t)off + 15;
1088 else
1089 pVCpu->iem.s.cbInstrBuf = pVCpu->iem.s.cbInstrBufTotal;
1090 }
1091 else
1092 {
1093 pVCpu->iem.s.pbInstrBuf = NULL;
1094 pVCpu->iem.s.offInstrNextByte = 0;
1095 pVCpu->iem.s.offCurInstrStart = 0;
1096 pVCpu->iem.s.cbInstrBuf = 0;
1097 pVCpu->iem.s.cbInstrBufTotal = 0;
1098 }
1099 }
1100 else
1101 {
1102 pVCpu->iem.s.offInstrNextByte = 0;
1103 pVCpu->iem.s.offCurInstrStart = 0;
1104 pVCpu->iem.s.cbInstrBuf = 0;
1105 pVCpu->iem.s.cbInstrBufTotal = 0;
1106 }
1107#else
1108 pVCpu->iem.s.cbOpcode = 0;
1109 pVCpu->iem.s.offOpcode = 0;
1110#endif
1111 Assert(pVCpu->iem.s.cActiveMappings == 0);
1112 pVCpu->iem.s.iNextMapping = 0;
1113 Assert(pVCpu->iem.s.rcPassUp == VINF_SUCCESS);
1114 Assert(pVCpu->iem.s.fBypassHandlers == false);
1115#ifdef VBOX_WITH_RAW_MODE_NOT_R0
1116 if (!pVCpu->iem.s.fInPatchCode)
1117 { /* likely */ }
1118 else
1119 {
1120 pVCpu->iem.s.fInPatchCode = pVCpu->iem.s.uCpl == 0
1121 && pCtx->cs.u64Base == 0
1122 && pCtx->cs.u32Limit == UINT32_MAX
1123 && PATMIsPatchGCAddr(pVCpu->CTX_SUFF(pVM), pCtx->eip);
1124 if (!pVCpu->iem.s.fInPatchCode)
1125 CPUMRawLeave(pVCpu, VINF_SUCCESS);
1126 }
1127#endif
1128
1129#ifdef DBGFTRACE_ENABLED
1130 switch (enmMode)
1131 {
1132 case IEMMODE_64BIT:
1133 RTTraceBufAddMsgF(pVCpu->CTX_SUFF(pVM)->CTX_SUFF(hTraceBuf), "I64/%u %08llx", pVCpu->iem.s.uCpl, pCtx->rip);
1134 break;
1135 case IEMMODE_32BIT:
1136 RTTraceBufAddMsgF(pVCpu->CTX_SUFF(pVM)->CTX_SUFF(hTraceBuf), "I32/%u %04x:%08x", pVCpu->iem.s.uCpl, pCtx->cs.Sel, pCtx->eip);
1137 break;
1138 case IEMMODE_16BIT:
1139 RTTraceBufAddMsgF(pVCpu->CTX_SUFF(pVM)->CTX_SUFF(hTraceBuf), "I16/%u %04x:%04x", pVCpu->iem.s.uCpl, pCtx->cs.Sel, pCtx->eip);
1140 break;
1141 }
1142#endif
1143}
1144
1145
1146
1147/**
1148 * Prefetch opcodes the first time when starting executing.
1149 *
1150 * @returns Strict VBox status code.
1151 * @param pVCpu The cross context virtual CPU structure of the
1152 * calling thread.
1153 * @param fBypassHandlers Whether to bypass access handlers.
1154 */
1155IEM_STATIC VBOXSTRICTRC iemInitDecoderAndPrefetchOpcodes(PVMCPU pVCpu, bool fBypassHandlers)
1156{
1157#ifdef IEM_VERIFICATION_MODE_FULL
1158 uint8_t const cbOldOpcodes = pVCpu->iem.s.cbOpcode;
1159#endif
1160 iemInitDecoder(pVCpu, fBypassHandlers);
1161
1162#ifdef IEM_WITH_CODE_TLB
1163 /** @todo Do ITLB lookup here. */
1164
1165#else /* !IEM_WITH_CODE_TLB */
1166
1167 /*
1168 * What we're doing here is very similar to iemMemMap/iemMemBounceBufferMap.
1169 *
1170 * First translate CS:rIP to a physical address.
1171 */
1172 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
1173 uint32_t cbToTryRead;
1174 RTGCPTR GCPtrPC;
1175 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
1176 {
1177 cbToTryRead = PAGE_SIZE;
1178 GCPtrPC = pCtx->rip;
1179 if (!IEM_IS_CANONICAL(GCPtrPC))
1180 return iemRaiseGeneralProtectionFault0(pVCpu);
1181 cbToTryRead = PAGE_SIZE - (GCPtrPC & PAGE_OFFSET_MASK);
1182 }
1183 else
1184 {
1185 uint32_t GCPtrPC32 = pCtx->eip;
1186 AssertMsg(!(GCPtrPC32 & ~(uint32_t)UINT16_MAX) || pVCpu->iem.s.enmCpuMode == IEMMODE_32BIT, ("%04x:%RX64\n", pCtx->cs.Sel, pCtx->rip));
1187 if (GCPtrPC32 > pCtx->cs.u32Limit)
1188 return iemRaiseSelectorBounds(pVCpu, X86_SREG_CS, IEM_ACCESS_INSTRUCTION);
1189 cbToTryRead = pCtx->cs.u32Limit - GCPtrPC32 + 1;
1190 if (!cbToTryRead) /* overflowed */
1191 {
1192 Assert(GCPtrPC32 == 0); Assert(pCtx->cs.u32Limit == UINT32_MAX);
1193 cbToTryRead = UINT32_MAX;
1194 }
1195 GCPtrPC = (uint32_t)pCtx->cs.u64Base + GCPtrPC32;
1196 Assert(GCPtrPC <= UINT32_MAX);
1197 }
1198
1199# ifdef VBOX_WITH_RAW_MODE_NOT_R0
1200 /* Allow interpretation of patch manager code blocks since they can for
1201 instance throw #PFs for perfectly good reasons. */
1202 if (pVCpu->iem.s.fInPatchCode)
1203 {
1204 size_t cbRead = 0;
1205 int rc = PATMReadPatchCode(pVCpu->CTX_SUFF(pVM), GCPtrPC, pVCpu->iem.s.abOpcode, sizeof(pVCpu->iem.s.abOpcode), &cbRead);
1206 AssertRCReturn(rc, rc);
1207 pVCpu->iem.s.cbOpcode = (uint8_t)cbRead; Assert(pVCpu->iem.s.cbOpcode == cbRead); Assert(cbRead > 0);
1208 return VINF_SUCCESS;
1209 }
1210# endif /* VBOX_WITH_RAW_MODE_NOT_R0 */
1211
1212 RTGCPHYS GCPhys;
1213 uint64_t fFlags;
1214 int rc = PGMGstGetPage(pVCpu, GCPtrPC, &fFlags, &GCPhys);
1215 if (RT_FAILURE(rc))
1216 {
1217 Log(("iemInitDecoderAndPrefetchOpcodes: %RGv - rc=%Rrc\n", GCPtrPC, rc));
1218 return iemRaisePageFault(pVCpu, GCPtrPC, IEM_ACCESS_INSTRUCTION, rc);
1219 }
1220 if (!(fFlags & X86_PTE_US) && pVCpu->iem.s.uCpl == 3)
1221 {
1222 Log(("iemInitDecoderAndPrefetchOpcodes: %RGv - supervisor page\n", GCPtrPC));
1223 return iemRaisePageFault(pVCpu, GCPtrPC, IEM_ACCESS_INSTRUCTION, VERR_ACCESS_DENIED);
1224 }
1225 if ((fFlags & X86_PTE_PAE_NX) && (pCtx->msrEFER & MSR_K6_EFER_NXE))
1226 {
1227 Log(("iemInitDecoderAndPrefetchOpcodes: %RGv - NX\n", GCPtrPC));
1228 return iemRaisePageFault(pVCpu, GCPtrPC, IEM_ACCESS_INSTRUCTION, VERR_ACCESS_DENIED);
1229 }
1230 GCPhys |= GCPtrPC & PAGE_OFFSET_MASK;
1231 /** @todo Check reserved bits and such stuff. PGM is better at doing
1232 * that, so do it when implementing the guest virtual address
1233 * TLB... */
1234
1235# ifdef IEM_VERIFICATION_MODE_FULL
1236 /*
1237 * Optimistic optimization: Use unconsumed opcode bytes from the previous
1238 * instruction.
1239 */
1240 /** @todo optimize this differently by not using PGMPhysRead. */
1241 RTGCPHYS const offPrevOpcodes = GCPhys - pVCpu->iem.s.GCPhysOpcodes;
1242 pVCpu->iem.s.GCPhysOpcodes = GCPhys;
1243 if ( offPrevOpcodes < cbOldOpcodes
1244 && PAGE_SIZE - (GCPhys & PAGE_OFFSET_MASK) > sizeof(pVCpu->iem.s.abOpcode))
1245 {
1246 uint8_t cbNew = cbOldOpcodes - (uint8_t)offPrevOpcodes;
1247 Assert(cbNew <= RT_ELEMENTS(pVCpu->iem.s.abOpcode));
1248 memmove(&pVCpu->iem.s.abOpcode[0], &pVCpu->iem.s.abOpcode[offPrevOpcodes], cbNew);
1249 pVCpu->iem.s.cbOpcode = cbNew;
1250 return VINF_SUCCESS;
1251 }
1252# endif
1253
1254 /*
1255 * Read the bytes at this address.
1256 */
1257 PVM pVM = pVCpu->CTX_SUFF(pVM);
1258# if defined(IN_RING3) && defined(VBOX_WITH_RAW_MODE_NOT_R0)
1259 size_t cbActual;
1260 if ( PATMIsEnabled(pVM)
1261 && RT_SUCCESS(PATMR3ReadOrgInstr(pVM, GCPtrPC, pVCpu->iem.s.abOpcode, sizeof(pVCpu->iem.s.abOpcode), &cbActual)))
1262 {
1263 Log4(("decode - Read %u unpatched bytes at %RGv\n", cbActual, GCPtrPC));
1264 Assert(cbActual > 0);
1265 pVCpu->iem.s.cbOpcode = (uint8_t)cbActual;
1266 }
1267 else
1268# endif
1269 {
1270 uint32_t cbLeftOnPage = PAGE_SIZE - (GCPtrPC & PAGE_OFFSET_MASK);
1271 if (cbToTryRead > cbLeftOnPage)
1272 cbToTryRead = cbLeftOnPage;
1273 if (cbToTryRead > sizeof(pVCpu->iem.s.abOpcode))
1274 cbToTryRead = sizeof(pVCpu->iem.s.abOpcode);
1275
1276 if (!pVCpu->iem.s.fBypassHandlers)
1277 {
1278 VBOXSTRICTRC rcStrict = PGMPhysRead(pVM, GCPhys, pVCpu->iem.s.abOpcode, cbToTryRead, PGMACCESSORIGIN_IEM);
1279 if (RT_LIKELY(rcStrict == VINF_SUCCESS))
1280 { /* likely */ }
1281 else if (PGM_PHYS_RW_IS_SUCCESS(rcStrict))
1282 {
1283 Log(("iemInitDecoderAndPrefetchOpcodes: %RGv/%RGp LB %#x - read status - rcStrict=%Rrc\n",
1284 GCPtrPC, GCPhys, VBOXSTRICTRC_VAL(rcStrict), cbToTryRead));
1285 rcStrict = iemSetPassUpStatus(pVCpu, rcStrict);
1286 }
1287 else
1288 {
1289 Log((RT_SUCCESS(rcStrict)
1290 ? "iemInitDecoderAndPrefetchOpcodes: %RGv/%RGp LB %#x - read status - rcStrict=%Rrc\n"
1291 : "iemInitDecoderAndPrefetchOpcodes: %RGv/%RGp LB %#x - read error - rcStrict=%Rrc (!!)\n",
1292 GCPtrPC, GCPhys, VBOXSTRICTRC_VAL(rcStrict), cbToTryRead));
1293 return rcStrict;
1294 }
1295 }
1296 else
1297 {
1298 rc = PGMPhysSimpleReadGCPhys(pVM, pVCpu->iem.s.abOpcode, GCPhys, cbToTryRead);
1299 if (RT_SUCCESS(rc))
1300 { /* likely */ }
1301 else
1302 {
1303 Log(("iemInitDecoderAndPrefetchOpcodes: %RGv/%RGp LB %#x - read error - rc=%Rrc (!!)\n",
1304 GCPtrPC, GCPhys, rc, cbToTryRead));
1305 return rc;
1306 }
1307 }
1308 pVCpu->iem.s.cbOpcode = cbToTryRead;
1309 }
1310#endif /* !IEM_WITH_CODE_TLB */
1311 return VINF_SUCCESS;
1312}
1313
1314
1315/**
1316 * Invalidates the IEM TLBs.
1317 *
1318 * This is called internally as well as by PGM when moving GC mappings.
1319 *
1320 * @returns
1321 * @param pVCpu The cross context virtual CPU structure of the calling
1322 * thread.
1323 * @param fVmm Set when PGM calls us with a remapping.
1324 */
1325VMM_INT_DECL(void) IEMTlbInvalidateAll(PVMCPU pVCpu, bool fVmm)
1326{
1327#ifdef IEM_WITH_CODE_TLB
1328 pVCpu->iem.s.cbInstrBufTotal = 0;
1329 pVCpu->iem.s.CodeTlb.uTlbRevision += IEMTLB_REVISION_INCR;
1330 if (pVCpu->iem.s.CodeTlb.uTlbRevision != 0)
1331 { /* very likely */ }
1332 else
1333 {
1334 pVCpu->iem.s.CodeTlb.uTlbRevision = IEMTLB_REVISION_INCR;
1335 unsigned i = RT_ELEMENTS(pVCpu->iem.s.CodeTlb.aEntries);
1336 while (i-- > 0)
1337 pVCpu->iem.s.CodeTlb.aEntries[i].uTag = 0;
1338 }
1339#endif
1340
1341#ifdef IEM_WITH_DATA_TLB
1342 pVCpu->iem.s.DataTlb.uTlbRevision += IEMTLB_REVISION_INCR;
1343 if (pVCpu->iem.s.DataTlb.uTlbRevision != 0)
1344 { /* very likely */ }
1345 else
1346 {
1347 pVCpu->iem.s.DataTlb.uTlbRevision = IEMTLB_REVISION_INCR;
1348 unsigned i = RT_ELEMENTS(pVCpu->iem.s.DataTlb.aEntries);
1349 while (i-- > 0)
1350 pVCpu->iem.s.DataTlb.aEntries[i].uTag = 0;
1351 }
1352#endif
1353 NOREF(pVCpu); NOREF(fVmm);
1354}
1355
1356
1357/**
1358 * Invalidates a page in the TLBs.
1359 *
1360 * @param pVCpu The cross context virtual CPU structure of the calling
1361 * thread.
1362 * @param GCPtr The address of the page to invalidate
1363 */
1364VMM_INT_DECL(void) IEMTlbInvalidatePage(PVMCPU pVCpu, RTGCPTR GCPtr)
1365{
1366#if defined(IEM_WITH_CODE_TLB) || defined(IEM_WITH_DATA_TLB)
1367 GCPtr = GCPtr >> X86_PAGE_SHIFT;
1368 AssertCompile(RT_ELEMENTS(pVCpu->iem.s.CodeTlb.aEntries) == 256);
1369 AssertCompile(RT_ELEMENTS(pVCpu->iem.s.DataTlb.aEntries) == 256);
1370 uintptr_t idx = (uint8_t)GCPtr;
1371
1372# ifdef IEM_WITH_CODE_TLB
1373 if (pVCpu->iem.s.CodeTlb.aEntries[idx].uTag == (GCPtr | pVCpu->iem.s.CodeTlb.uTlbRevision))
1374 {
1375 pVCpu->iem.s.CodeTlb.aEntries[idx].uTag = 0;
1376 if (GCPtr == (pVCpu->iem.s.uInstrBufPc >> X86_PAGE_SHIFT))
1377 pVCpu->iem.s.cbInstrBufTotal = 0;
1378 }
1379# endif
1380
1381# ifdef IEM_WITH_DATA_TLB
1382 if (pVCpu->iem.s.DataTlb.aEntries[idx].uTag == (GCPtr | pVCpu->iem.s.DataTlb.uTlbRevision))
1383 pVCpu->iem.s.DataTlb.aEntries[idx].uTag = 0;
1384# endif
1385#else
1386 NOREF(pVCpu); NOREF(GCPtr);
1387#endif
1388}
1389
1390
1391/**
1392 * Invalidates the host physical aspects of the IEM TLBs.
1393 *
1394 * This is called internally as well as by PGM when moving GC mappings.
1395 *
1396 * @param pVCpu The cross context virtual CPU structure of the calling
1397 * thread.
1398 */
1399VMM_INT_DECL(void) IEMTlbInvalidateAllPhysical(PVMCPU pVCpu)
1400{
1401#if defined(IEM_WITH_CODE_TLB) || defined(IEM_WITH_DATA_TLB)
1402 /* Note! This probably won't end up looking exactly like this, but it give an idea... */
1403
1404# ifdef IEM_WITH_CODE_TLB
1405 pVCpu->iem.s.cbInstrBufTotal = 0;
1406# endif
1407 uint64_t uTlbPhysRev = pVCpu->iem.s.CodeTlb.uTlbPhysRev + IEMTLB_PHYS_REV_INCR;
1408 if (uTlbPhysRev != 0)
1409 {
1410 pVCpu->iem.s.CodeTlb.uTlbPhysRev = uTlbPhysRev;
1411 pVCpu->iem.s.DataTlb.uTlbPhysRev = uTlbPhysRev;
1412 }
1413 else
1414 {
1415 pVCpu->iem.s.CodeTlb.uTlbPhysRev = IEMTLB_PHYS_REV_INCR;
1416 pVCpu->iem.s.DataTlb.uTlbPhysRev = IEMTLB_PHYS_REV_INCR;
1417
1418 unsigned i;
1419# ifdef IEM_WITH_CODE_TLB
1420 i = RT_ELEMENTS(pVCpu->iem.s.CodeTlb.aEntries);
1421 while (i-- > 0)
1422 {
1423 pVCpu->iem.s.CodeTlb.aEntries[i].pbMappingR3 = NULL;
1424 pVCpu->iem.s.CodeTlb.aEntries[i].fFlagsAndPhysRev &= ~(IEMTLBE_F_PG_NO_WRITE | IEMTLBE_F_PG_NO_READ | IEMTLBE_F_PHYS_REV);
1425 }
1426# endif
1427# ifdef IEM_WITH_DATA_TLB
1428 i = RT_ELEMENTS(pVCpu->iem.s.DataTlb.aEntries);
1429 while (i-- > 0)
1430 {
1431 pVCpu->iem.s.DataTlb.aEntries[i].pbMappingR3 = NULL;
1432 pVCpu->iem.s.DataTlb.aEntries[i].fFlagsAndPhysRev &= ~(IEMTLBE_F_PG_NO_WRITE | IEMTLBE_F_PG_NO_READ | IEMTLBE_F_PHYS_REV);
1433 }
1434# endif
1435 }
1436#else
1437 NOREF(pVCpu);
1438#endif
1439}
1440
1441
1442/**
1443 * Invalidates the host physical aspects of the IEM TLBs.
1444 *
1445 * This is called internally as well as by PGM when moving GC mappings.
1446 *
1447 * @param pVM The cross context VM structure.
1448 *
1449 * @remarks Caller holds the PGM lock.
1450 */
1451VMM_INT_DECL(void) IEMTlbInvalidateAllPhysicalAllCpus(PVM pVM)
1452{
1453 RT_NOREF_PV(pVM);
1454}
1455
1456#ifdef IEM_WITH_CODE_TLB
1457
1458/**
1459 * Tries to fetches @a cbDst opcode bytes, raise the appropriate exception on
1460 * failure and jumps.
1461 *
1462 * We end up here for a number of reasons:
1463 * - pbInstrBuf isn't yet initialized.
1464 * - Advancing beyond the buffer boundrary (e.g. cross page).
1465 * - Advancing beyond the CS segment limit.
1466 * - Fetching from non-mappable page (e.g. MMIO).
1467 *
1468 * @param pVCpu The cross context virtual CPU structure of the
1469 * calling thread.
1470 * @param pvDst Where to return the bytes.
1471 * @param cbDst Number of bytes to read.
1472 *
1473 * @todo Make cbDst = 0 a way of initializing pbInstrBuf?
1474 */
1475IEM_STATIC void iemOpcodeFetchBytesJmp(PVMCPU pVCpu, size_t cbDst, void *pvDst)
1476{
1477#ifdef IN_RING3
1478//__debugbreak();
1479#else
1480 longjmp(*CTX_SUFF(pVCpu->iem.s.pJmpBuf), VERR_INTERNAL_ERROR);
1481#endif
1482 for (;;)
1483 {
1484 Assert(cbDst <= 8);
1485 uint32_t offBuf = pVCpu->iem.s.offInstrNextByte;
1486
1487 /*
1488 * We might have a partial buffer match, deal with that first to make the
1489 * rest simpler. This is the first part of the cross page/buffer case.
1490 */
1491 if (pVCpu->iem.s.pbInstrBuf != NULL)
1492 {
1493 if (offBuf < pVCpu->iem.s.cbInstrBuf)
1494 {
1495 Assert(offBuf + cbDst > pVCpu->iem.s.cbInstrBuf);
1496 uint32_t const cbCopy = pVCpu->iem.s.cbInstrBuf - pVCpu->iem.s.offInstrNextByte;
1497 memcpy(pvDst, &pVCpu->iem.s.pbInstrBuf[offBuf], cbCopy);
1498
1499 cbDst -= cbCopy;
1500 pvDst = (uint8_t *)pvDst + cbCopy;
1501 offBuf += cbCopy;
1502 pVCpu->iem.s.offInstrNextByte += offBuf;
1503 }
1504 }
1505
1506 /*
1507 * Check segment limit, figuring how much we're allowed to access at this point.
1508 *
1509 * We will fault immediately if RIP is past the segment limit / in non-canonical
1510 * territory. If we do continue, there are one or more bytes to read before we
1511 * end up in trouble and we need to do that first before faulting.
1512 */
1513 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
1514 RTGCPTR GCPtrFirst;
1515 uint32_t cbMaxRead;
1516 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
1517 {
1518 GCPtrFirst = pCtx->rip + (offBuf - (uint32_t)(int32_t)pVCpu->iem.s.offCurInstrStart);
1519 if (RT_LIKELY(IEM_IS_CANONICAL(GCPtrFirst)))
1520 { /* likely */ }
1521 else
1522 iemRaiseGeneralProtectionFault0Jmp(pVCpu);
1523 cbMaxRead = X86_PAGE_SIZE - ((uint32_t)GCPtrFirst & X86_PAGE_OFFSET_MASK);
1524 }
1525 else
1526 {
1527 GCPtrFirst = pCtx->eip + (offBuf - (uint32_t)(int32_t)pVCpu->iem.s.offCurInstrStart);
1528 Assert(!(GCPtrFirst & ~(uint32_t)UINT16_MAX) || pVCpu->iem.s.enmCpuMode == IEMMODE_32BIT);
1529 if (RT_LIKELY((uint32_t)GCPtrFirst <= pCtx->cs.u32Limit))
1530 { /* likely */ }
1531 else
1532 iemRaiseSelectorBoundsJmp(pVCpu, X86_SREG_CS, IEM_ACCESS_INSTRUCTION);
1533 cbMaxRead = pCtx->cs.u32Limit - (uint32_t)GCPtrFirst + 1;
1534 if (cbMaxRead != 0)
1535 { /* likely */ }
1536 else
1537 {
1538 /* Overflowed because address is 0 and limit is max. */
1539 Assert(GCPtrFirst == 0); Assert(pCtx->cs.u32Limit == UINT32_MAX);
1540 cbMaxRead = X86_PAGE_SIZE;
1541 }
1542 GCPtrFirst = (uint32_t)GCPtrFirst + (uint32_t)pCtx->cs.u64Base;
1543 uint32_t cbMaxRead2 = X86_PAGE_SIZE - ((uint32_t)GCPtrFirst & X86_PAGE_OFFSET_MASK);
1544 if (cbMaxRead2 < cbMaxRead)
1545 cbMaxRead = cbMaxRead2;
1546 /** @todo testcase: unreal modes, both huge 16-bit and 32-bit. */
1547 }
1548
1549 /*
1550 * Get the TLB entry for this piece of code.
1551 */
1552 uint64_t uTag = (GCPtrFirst >> X86_PAGE_SHIFT) | pVCpu->iem.s.CodeTlb.uTlbRevision;
1553 AssertCompile(RT_ELEMENTS(pVCpu->iem.s.CodeTlb.aEntries) == 256);
1554 PIEMTLBENTRY pTlbe = &pVCpu->iem.s.CodeTlb.aEntries[(uint8_t)uTag];
1555 if (pTlbe->uTag == uTag)
1556 {
1557 /* likely when executing lots of code, otherwise unlikely */
1558# ifdef VBOX_WITH_STATISTICS
1559 pVCpu->iem.s.CodeTlb.cTlbHits++;
1560# endif
1561 }
1562 else
1563 {
1564 pVCpu->iem.s.CodeTlb.cTlbMisses++;
1565# ifdef VBOX_WITH_RAW_MODE_NOT_R0
1566 if (PATMIsPatchGCAddr(pVCpu->CTX_SUFF(pVM), pCtx->eip))
1567 {
1568 pTlbe->uTag = uTag;
1569 pTlbe->fFlagsAndPhysRev = IEMTLBE_F_PATCH_CODE | IEMTLBE_F_PT_NO_WRITE | IEMTLBE_F_PT_NO_USER
1570 | IEMTLBE_F_PT_NO_WRITE | IEMTLBE_F_PT_NO_DIRTY | IEMTLBE_F_NO_MAPPINGR3;
1571 pTlbe->GCPhys = NIL_RTGCPHYS;
1572 pTlbe->pbMappingR3 = NULL;
1573 }
1574 else
1575# endif
1576 {
1577 RTGCPHYS GCPhys;
1578 uint64_t fFlags;
1579 int rc = PGMGstGetPage(pVCpu, GCPtrFirst, &fFlags, &GCPhys);
1580 if (RT_FAILURE(rc))
1581 {
1582 Log(("iemOpcodeFetchMoreBytes: %RGv - rc=%Rrc\n", GCPtrFirst, rc));
1583 iemRaisePageFaultJmp(pVCpu, GCPtrFirst, IEM_ACCESS_INSTRUCTION, rc);
1584 }
1585
1586 AssertCompile(IEMTLBE_F_PT_NO_EXEC == 1);
1587 pTlbe->uTag = uTag;
1588 pTlbe->fFlagsAndPhysRev = (~fFlags & (X86_PTE_US | X86_PTE_RW | X86_PTE_D)) | (fFlags >> X86_PTE_PAE_BIT_NX);
1589 pTlbe->GCPhys = GCPhys;
1590 pTlbe->pbMappingR3 = NULL;
1591 }
1592 }
1593
1594 /*
1595 * Check TLB page table level access flags.
1596 */
1597 if (pTlbe->fFlagsAndPhysRev & (IEMTLBE_F_PT_NO_USER | IEMTLBE_F_PT_NO_EXEC))
1598 {
1599 if ((pTlbe->fFlagsAndPhysRev & IEMTLBE_F_PT_NO_USER) && pVCpu->iem.s.uCpl == 3)
1600 {
1601 Log(("iemOpcodeFetchBytesJmp: %RGv - supervisor page\n", GCPtrFirst));
1602 iemRaisePageFaultJmp(pVCpu, GCPtrFirst, IEM_ACCESS_INSTRUCTION, VERR_ACCESS_DENIED);
1603 }
1604 if ((pTlbe->fFlagsAndPhysRev & IEMTLBE_F_PT_NO_EXEC) && (pCtx->msrEFER & MSR_K6_EFER_NXE))
1605 {
1606 Log(("iemOpcodeFetchMoreBytes: %RGv - NX\n", GCPtrFirst));
1607 iemRaisePageFaultJmp(pVCpu, GCPtrFirst, IEM_ACCESS_INSTRUCTION, VERR_ACCESS_DENIED);
1608 }
1609 }
1610
1611# ifdef VBOX_WITH_RAW_MODE_NOT_R0
1612 /*
1613 * Allow interpretation of patch manager code blocks since they can for
1614 * instance throw #PFs for perfectly good reasons.
1615 */
1616 if (!(pTlbe->fFlagsAndPhysRev & IEMTLBE_F_PATCH_CODE))
1617 { /* no unlikely */ }
1618 else
1619 {
1620 /** @todo Could be optimized this a little in ring-3 if we liked. */
1621 size_t cbRead = 0;
1622 int rc = PATMReadPatchCode(pVCpu->CTX_SUFF(pVM), GCPtrFirst, pvDst, cbDst, &cbRead);
1623 AssertRCStmt(rc, longjmp(*CTX_SUFF(pVCpu->iem.s.pJmpBuf), rc));
1624 AssertStmt(cbRead == cbDst, longjmp(*CTX_SUFF(pVCpu->iem.s.pJmpBuf), VERR_IEM_IPE_1));
1625 return;
1626 }
1627# endif /* VBOX_WITH_RAW_MODE_NOT_R0 */
1628
1629 /*
1630 * Look up the physical page info if necessary.
1631 */
1632 if ((pTlbe->fFlagsAndPhysRev & IEMTLBE_F_PHYS_REV) == pVCpu->iem.s.CodeTlb.uTlbPhysRev)
1633 { /* not necessary */ }
1634 else
1635 {
1636 AssertCompile(PGMIEMGCPHYS2PTR_F_NO_WRITE == IEMTLBE_F_PG_NO_WRITE);
1637 AssertCompile(PGMIEMGCPHYS2PTR_F_NO_READ == IEMTLBE_F_PG_NO_READ);
1638 AssertCompile(PGMIEMGCPHYS2PTR_F_NO_MAPPINGR3 == IEMTLBE_F_NO_MAPPINGR3);
1639 pTlbe->fFlagsAndPhysRev &= ~( IEMTLBE_F_PHYS_REV
1640 | IEMTLBE_F_NO_MAPPINGR3 | IEMTLBE_F_PG_NO_READ | IEMTLBE_F_PG_NO_WRITE);
1641 int rc = PGMPhysIemGCPhys2PtrNoLock(pVCpu->CTX_SUFF(pVM), pVCpu, pTlbe->GCPhys, &pVCpu->iem.s.CodeTlb.uTlbPhysRev,
1642 &pTlbe->pbMappingR3, &pTlbe->fFlagsAndPhysRev);
1643 AssertRCStmt(rc, longjmp(*CTX_SUFF(pVCpu->iem.s.pJmpBuf), rc));
1644 }
1645
1646# if defined(IN_RING3) || (defined(IN_RING0) && !defined(VBOX_WITH_2X_4GB_ADDR_SPACE))
1647 /*
1648 * Try do a direct read using the pbMappingR3 pointer.
1649 */
1650 if ( (pTlbe->fFlagsAndPhysRev & (IEMTLBE_F_PHYS_REV | IEMTLBE_F_NO_MAPPINGR3 | IEMTLBE_F_PG_NO_READ))
1651 == pVCpu->iem.s.CodeTlb.uTlbPhysRev)
1652 {
1653 uint32_t const offPg = (GCPtrFirst & X86_PAGE_OFFSET_MASK);
1654 pVCpu->iem.s.cbInstrBufTotal = offPg + cbMaxRead;
1655 if (offBuf == (uint32_t)(int32_t)pVCpu->iem.s.offCurInstrStart)
1656 {
1657 pVCpu->iem.s.cbInstrBuf = offPg + RT_MIN(15, cbMaxRead);
1658 pVCpu->iem.s.offCurInstrStart = (int16_t)offPg;
1659 }
1660 else
1661 {
1662 uint32_t const cbInstr = offBuf - (uint32_t)(int32_t)pVCpu->iem.s.offCurInstrStart;
1663 Assert(cbInstr < cbMaxRead);
1664 pVCpu->iem.s.cbInstrBuf = offPg + RT_MIN(cbMaxRead + cbInstr, 15) - cbInstr;
1665 pVCpu->iem.s.offCurInstrStart = (int16_t)(offPg - cbInstr);
1666 }
1667 if (cbDst <= cbMaxRead)
1668 {
1669 pVCpu->iem.s.offInstrNextByte = offPg + (uint32_t)cbDst;
1670 pVCpu->iem.s.uInstrBufPc = GCPtrFirst & ~(RTGCPTR)X86_PAGE_OFFSET_MASK;
1671 pVCpu->iem.s.pbInstrBuf = pTlbe->pbMappingR3;
1672 memcpy(pvDst, &pTlbe->pbMappingR3[offPg], cbDst);
1673 return;
1674 }
1675 pVCpu->iem.s.pbInstrBuf = NULL;
1676
1677 memcpy(pvDst, &pTlbe->pbMappingR3[offPg], cbMaxRead);
1678 pVCpu->iem.s.offInstrNextByte = offPg + cbMaxRead;
1679 }
1680 else
1681# endif
1682#if 0
1683 /*
1684 * If there is no special read handling, so we can read a bit more and
1685 * put it in the prefetch buffer.
1686 */
1687 if ( cbDst < cbMaxRead
1688 && (pTlbe->fFlagsAndPhysRev & (IEMTLBE_F_PHYS_REV | IEMTLBE_F_PG_NO_READ)) == pVCpu->iem.s.CodeTlb.uTlbPhysRev)
1689 {
1690 VBOXSTRICTRC rcStrict = PGMPhysRead(pVCpu->CTX_SUFF(pVM), pTlbe->GCPhys,
1691 &pVCpu->iem.s.abOpcode[0], cbToTryRead, PGMACCESSORIGIN_IEM);
1692 if (RT_LIKELY(rcStrict == VINF_SUCCESS))
1693 { /* likely */ }
1694 else if (PGM_PHYS_RW_IS_SUCCESS(rcStrict))
1695 {
1696 Log(("iemOpcodeFetchMoreBytes: %RGv/%RGp LB %#x - read status - rcStrict=%Rrc\n",
1697 GCPtrNext, GCPhys, VBOXSTRICTRC_VAL(rcStrict), cbToTryRead));
1698 rcStrict = iemSetPassUpStatus(pVCpu, rcStrict);
1699 AssertStmt(rcStrict == VINF_SUCCESS, longjmp(*CTX_SUFF(pVCpu->iem.s.pJmpBuf), VBOXSTRICRC_VAL(rcStrict)));
1700 }
1701 else
1702 {
1703 Log((RT_SUCCESS(rcStrict)
1704 ? "iemOpcodeFetchMoreBytes: %RGv/%RGp LB %#x - read status - rcStrict=%Rrc\n"
1705 : "iemOpcodeFetchMoreBytes: %RGv/%RGp LB %#x - read error - rcStrict=%Rrc (!!)\n",
1706 GCPtrNext, GCPhys, VBOXSTRICTRC_VAL(rcStrict), cbToTryRead));
1707 longjmp(*CTX_SUFF(pVCpu->iem.s.pJmpBuf), VBOXSTRICTRC_VAL(rcStrict));
1708 }
1709 }
1710 /*
1711 * Special read handling, so only read exactly what's needed.
1712 * This is a highly unlikely scenario.
1713 */
1714 else
1715#endif
1716 {
1717 pVCpu->iem.s.CodeTlb.cTlbSlowReadPath++;
1718 uint32_t const cbToRead = RT_MIN((uint32_t)cbDst, cbMaxRead);
1719 VBOXSTRICTRC rcStrict = PGMPhysRead(pVCpu->CTX_SUFF(pVM), pTlbe->GCPhys + (GCPtrFirst & X86_PAGE_OFFSET_MASK),
1720 pvDst, cbToRead, PGMACCESSORIGIN_IEM);
1721 if (RT_LIKELY(rcStrict == VINF_SUCCESS))
1722 { /* likely */ }
1723 else if (PGM_PHYS_RW_IS_SUCCESS(rcStrict))
1724 {
1725 Log(("iemOpcodeFetchMoreBytes: %RGv/%RGp LB %#x - read status - rcStrict=%Rrc\n",
1726 GCPtrFirst, pTlbe->GCPhys + (GCPtrFirst & X86_PAGE_OFFSET_MASK), VBOXSTRICTRC_VAL(rcStrict), cbToRead));
1727 rcStrict = iemSetPassUpStatus(pVCpu, rcStrict);
1728 AssertStmt(rcStrict == VINF_SUCCESS, longjmp(*CTX_SUFF(pVCpu->iem.s.pJmpBuf), VBOXSTRICTRC_VAL(rcStrict)));
1729 }
1730 else
1731 {
1732 Log((RT_SUCCESS(rcStrict)
1733 ? "iemOpcodeFetchMoreBytes: %RGv/%RGp LB %#x - read status - rcStrict=%Rrc\n"
1734 : "iemOpcodeFetchMoreBytes: %RGv/%RGp LB %#x - read error - rcStrict=%Rrc (!!)\n",
1735 GCPtrFirst, pTlbe->GCPhys + (GCPtrFirst & X86_PAGE_OFFSET_MASK), VBOXSTRICTRC_VAL(rcStrict), cbToRead));
1736 longjmp(*CTX_SUFF(pVCpu->iem.s.pJmpBuf), VBOXSTRICTRC_VAL(rcStrict));
1737 }
1738 pVCpu->iem.s.offInstrNextByte = offBuf + cbToRead;
1739 if (cbToRead == cbDst)
1740 return;
1741 }
1742
1743 /*
1744 * More to read, loop.
1745 */
1746 cbDst -= cbMaxRead;
1747 pvDst = (uint8_t *)pvDst + cbMaxRead;
1748 }
1749}
1750
1751#else
1752
1753/**
1754 * Try fetch at least @a cbMin bytes more opcodes, raise the appropriate
1755 * exception if it fails.
1756 *
1757 * @returns Strict VBox status code.
1758 * @param pVCpu The cross context virtual CPU structure of the
1759 * calling thread.
1760 * @param cbMin The minimum number of bytes relative offOpcode
1761 * that must be read.
1762 */
1763IEM_STATIC VBOXSTRICTRC iemOpcodeFetchMoreBytes(PVMCPU pVCpu, size_t cbMin)
1764{
1765 /*
1766 * What we're doing here is very similar to iemMemMap/iemMemBounceBufferMap.
1767 *
1768 * First translate CS:rIP to a physical address.
1769 */
1770 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
1771 uint8_t cbLeft = pVCpu->iem.s.cbOpcode - pVCpu->iem.s.offOpcode; Assert(cbLeft < cbMin);
1772 uint32_t cbToTryRead;
1773 RTGCPTR GCPtrNext;
1774 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
1775 {
1776 cbToTryRead = PAGE_SIZE;
1777 GCPtrNext = pCtx->rip + pVCpu->iem.s.cbOpcode;
1778 if (!IEM_IS_CANONICAL(GCPtrNext))
1779 return iemRaiseGeneralProtectionFault0(pVCpu);
1780 }
1781 else
1782 {
1783 uint32_t GCPtrNext32 = pCtx->eip;
1784 Assert(!(GCPtrNext32 & ~(uint32_t)UINT16_MAX) || pVCpu->iem.s.enmCpuMode == IEMMODE_32BIT);
1785 GCPtrNext32 += pVCpu->iem.s.cbOpcode;
1786 if (GCPtrNext32 > pCtx->cs.u32Limit)
1787 return iemRaiseSelectorBounds(pVCpu, X86_SREG_CS, IEM_ACCESS_INSTRUCTION);
1788 cbToTryRead = pCtx->cs.u32Limit - GCPtrNext32 + 1;
1789 if (!cbToTryRead) /* overflowed */
1790 {
1791 Assert(GCPtrNext32 == 0); Assert(pCtx->cs.u32Limit == UINT32_MAX);
1792 cbToTryRead = UINT32_MAX;
1793 /** @todo check out wrapping around the code segment. */
1794 }
1795 if (cbToTryRead < cbMin - cbLeft)
1796 return iemRaiseSelectorBounds(pVCpu, X86_SREG_CS, IEM_ACCESS_INSTRUCTION);
1797 GCPtrNext = (uint32_t)pCtx->cs.u64Base + GCPtrNext32;
1798 }
1799
1800 /* Only read up to the end of the page, and make sure we don't read more
1801 than the opcode buffer can hold. */
1802 uint32_t cbLeftOnPage = PAGE_SIZE - (GCPtrNext & PAGE_OFFSET_MASK);
1803 if (cbToTryRead > cbLeftOnPage)
1804 cbToTryRead = cbLeftOnPage;
1805 if (cbToTryRead > sizeof(pVCpu->iem.s.abOpcode) - pVCpu->iem.s.cbOpcode)
1806 cbToTryRead = sizeof(pVCpu->iem.s.abOpcode) - pVCpu->iem.s.cbOpcode;
1807/** @todo r=bird: Convert assertion into undefined opcode exception? */
1808 Assert(cbToTryRead >= cbMin - cbLeft); /* ASSUMPTION based on iemInitDecoderAndPrefetchOpcodes. */
1809
1810# ifdef VBOX_WITH_RAW_MODE_NOT_R0
1811 /* Allow interpretation of patch manager code blocks since they can for
1812 instance throw #PFs for perfectly good reasons. */
1813 if (pVCpu->iem.s.fInPatchCode)
1814 {
1815 size_t cbRead = 0;
1816 int rc = PATMReadPatchCode(pVCpu->CTX_SUFF(pVM), GCPtrNext, pVCpu->iem.s.abOpcode, cbToTryRead, &cbRead);
1817 AssertRCReturn(rc, rc);
1818 pVCpu->iem.s.cbOpcode = (uint8_t)cbRead; Assert(pVCpu->iem.s.cbOpcode == cbRead); Assert(cbRead > 0);
1819 return VINF_SUCCESS;
1820 }
1821# endif /* VBOX_WITH_RAW_MODE_NOT_R0 */
1822
1823 RTGCPHYS GCPhys;
1824 uint64_t fFlags;
1825 int rc = PGMGstGetPage(pVCpu, GCPtrNext, &fFlags, &GCPhys);
1826 if (RT_FAILURE(rc))
1827 {
1828 Log(("iemOpcodeFetchMoreBytes: %RGv - rc=%Rrc\n", GCPtrNext, rc));
1829 return iemRaisePageFault(pVCpu, GCPtrNext, IEM_ACCESS_INSTRUCTION, rc);
1830 }
1831 if (!(fFlags & X86_PTE_US) && pVCpu->iem.s.uCpl == 3)
1832 {
1833 Log(("iemOpcodeFetchMoreBytes: %RGv - supervisor page\n", GCPtrNext));
1834 return iemRaisePageFault(pVCpu, GCPtrNext, IEM_ACCESS_INSTRUCTION, VERR_ACCESS_DENIED);
1835 }
1836 if ((fFlags & X86_PTE_PAE_NX) && (pCtx->msrEFER & MSR_K6_EFER_NXE))
1837 {
1838 Log(("iemOpcodeFetchMoreBytes: %RGv - NX\n", GCPtrNext));
1839 return iemRaisePageFault(pVCpu, GCPtrNext, IEM_ACCESS_INSTRUCTION, VERR_ACCESS_DENIED);
1840 }
1841 GCPhys |= GCPtrNext & PAGE_OFFSET_MASK;
1842 Log5(("GCPtrNext=%RGv GCPhys=%RGp cbOpcodes=%#x\n", GCPtrNext, GCPhys, pVCpu->iem.s.cbOpcode));
1843 /** @todo Check reserved bits and such stuff. PGM is better at doing
1844 * that, so do it when implementing the guest virtual address
1845 * TLB... */
1846
1847 /*
1848 * Read the bytes at this address.
1849 *
1850 * We read all unpatched bytes in iemInitDecoderAndPrefetchOpcodes already,
1851 * and since PATM should only patch the start of an instruction there
1852 * should be no need to check again here.
1853 */
1854 if (!pVCpu->iem.s.fBypassHandlers)
1855 {
1856 VBOXSTRICTRC rcStrict = PGMPhysRead(pVCpu->CTX_SUFF(pVM), GCPhys, &pVCpu->iem.s.abOpcode[pVCpu->iem.s.cbOpcode],
1857 cbToTryRead, PGMACCESSORIGIN_IEM);
1858 if (RT_LIKELY(rcStrict == VINF_SUCCESS))
1859 { /* likely */ }
1860 else if (PGM_PHYS_RW_IS_SUCCESS(rcStrict))
1861 {
1862 Log(("iemOpcodeFetchMoreBytes: %RGv/%RGp LB %#x - read status - rcStrict=%Rrc\n",
1863 GCPtrNext, GCPhys, VBOXSTRICTRC_VAL(rcStrict), cbToTryRead));
1864 rcStrict = iemSetPassUpStatus(pVCpu, rcStrict);
1865 }
1866 else
1867 {
1868 Log((RT_SUCCESS(rcStrict)
1869 ? "iemOpcodeFetchMoreBytes: %RGv/%RGp LB %#x - read status - rcStrict=%Rrc\n"
1870 : "iemOpcodeFetchMoreBytes: %RGv/%RGp LB %#x - read error - rcStrict=%Rrc (!!)\n",
1871 GCPtrNext, GCPhys, VBOXSTRICTRC_VAL(rcStrict), cbToTryRead));
1872 return rcStrict;
1873 }
1874 }
1875 else
1876 {
1877 rc = PGMPhysSimpleReadGCPhys(pVCpu->CTX_SUFF(pVM), &pVCpu->iem.s.abOpcode[pVCpu->iem.s.cbOpcode], GCPhys, cbToTryRead);
1878 if (RT_SUCCESS(rc))
1879 { /* likely */ }
1880 else
1881 {
1882 Log(("iemOpcodeFetchMoreBytes: %RGv - read error - rc=%Rrc (!!)\n", GCPtrNext, rc));
1883 return rc;
1884 }
1885 }
1886 pVCpu->iem.s.cbOpcode += cbToTryRead;
1887 Log5(("%.*Rhxs\n", pVCpu->iem.s.cbOpcode, pVCpu->iem.s.abOpcode));
1888
1889 return VINF_SUCCESS;
1890}
1891
1892#endif /* !IEM_WITH_CODE_TLB */
1893#ifndef IEM_WITH_SETJMP
1894
1895/**
1896 * Deals with the problematic cases that iemOpcodeGetNextU8 doesn't like.
1897 *
1898 * @returns Strict VBox status code.
1899 * @param pVCpu The cross context virtual CPU structure of the
1900 * calling thread.
1901 * @param pb Where to return the opcode byte.
1902 */
1903DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemOpcodeGetNextU8Slow(PVMCPU pVCpu, uint8_t *pb)
1904{
1905 VBOXSTRICTRC rcStrict = iemOpcodeFetchMoreBytes(pVCpu, 1);
1906 if (rcStrict == VINF_SUCCESS)
1907 {
1908 uint8_t offOpcode = pVCpu->iem.s.offOpcode;
1909 *pb = pVCpu->iem.s.abOpcode[offOpcode];
1910 pVCpu->iem.s.offOpcode = offOpcode + 1;
1911 }
1912 else
1913 *pb = 0;
1914 return rcStrict;
1915}
1916
1917
1918/**
1919 * Fetches the next opcode byte.
1920 *
1921 * @returns Strict VBox status code.
1922 * @param pVCpu The cross context virtual CPU structure of the
1923 * calling thread.
1924 * @param pu8 Where to return the opcode byte.
1925 */
1926DECLINLINE(VBOXSTRICTRC) iemOpcodeGetNextU8(PVMCPU pVCpu, uint8_t *pu8)
1927{
1928 uintptr_t const offOpcode = pVCpu->iem.s.offOpcode;
1929 if (RT_LIKELY((uint8_t)offOpcode < pVCpu->iem.s.cbOpcode))
1930 {
1931 pVCpu->iem.s.offOpcode = (uint8_t)offOpcode + 1;
1932 *pu8 = pVCpu->iem.s.abOpcode[offOpcode];
1933 return VINF_SUCCESS;
1934 }
1935 return iemOpcodeGetNextU8Slow(pVCpu, pu8);
1936}
1937
1938#else /* IEM_WITH_SETJMP */
1939
1940/**
1941 * Deals with the problematic cases that iemOpcodeGetNextU8Jmp doesn't like, longjmp on error.
1942 *
1943 * @returns The opcode byte.
1944 * @param pVCpu The cross context virtual CPU structure of the calling thread.
1945 */
1946DECL_NO_INLINE(IEM_STATIC, uint8_t) iemOpcodeGetNextU8SlowJmp(PVMCPU pVCpu)
1947{
1948# ifdef IEM_WITH_CODE_TLB
1949 uint8_t u8;
1950 iemOpcodeFetchBytesJmp(pVCpu, sizeof(u8), &u8);
1951 return u8;
1952# else
1953 VBOXSTRICTRC rcStrict = iemOpcodeFetchMoreBytes(pVCpu, 1);
1954 if (rcStrict == VINF_SUCCESS)
1955 return pVCpu->iem.s.abOpcode[pVCpu->iem.s.offOpcode++];
1956 longjmp(*pVCpu->iem.s.CTX_SUFF(pJmpBuf), VBOXSTRICTRC_VAL(rcStrict));
1957# endif
1958}
1959
1960
1961/**
1962 * Fetches the next opcode byte, longjmp on error.
1963 *
1964 * @returns The opcode byte.
1965 * @param pVCpu The cross context virtual CPU structure of the calling thread.
1966 */
1967DECLINLINE(uint8_t) iemOpcodeGetNextU8Jmp(PVMCPU pVCpu)
1968{
1969# ifdef IEM_WITH_CODE_TLB
1970 uintptr_t offBuf = pVCpu->iem.s.offInstrNextByte;
1971 uint8_t const *pbBuf = pVCpu->iem.s.pbInstrBuf;
1972 if (RT_LIKELY( pbBuf != NULL
1973 && offBuf < pVCpu->iem.s.cbInstrBuf))
1974 {
1975 pVCpu->iem.s.offInstrNextByte = (uint32_t)offBuf + 1;
1976 return pbBuf[offBuf];
1977 }
1978# else
1979 uintptr_t offOpcode = pVCpu->iem.s.offOpcode;
1980 if (RT_LIKELY((uint8_t)offOpcode < pVCpu->iem.s.cbOpcode))
1981 {
1982 pVCpu->iem.s.offOpcode = (uint8_t)offOpcode + 1;
1983 return pVCpu->iem.s.abOpcode[offOpcode];
1984 }
1985# endif
1986 return iemOpcodeGetNextU8SlowJmp(pVCpu);
1987}
1988
1989#endif /* IEM_WITH_SETJMP */
1990
1991/**
1992 * Fetches the next opcode byte, returns automatically on failure.
1993 *
1994 * @param a_pu8 Where to return the opcode byte.
1995 * @remark Implicitly references pVCpu.
1996 */
1997#ifndef IEM_WITH_SETJMP
1998# define IEM_OPCODE_GET_NEXT_U8(a_pu8) \
1999 do \
2000 { \
2001 VBOXSTRICTRC rcStrict2 = iemOpcodeGetNextU8(pVCpu, (a_pu8)); \
2002 if (rcStrict2 == VINF_SUCCESS) \
2003 { /* likely */ } \
2004 else \
2005 return rcStrict2; \
2006 } while (0)
2007#else
2008# define IEM_OPCODE_GET_NEXT_U8(a_pu8) (*(a_pu8) = iemOpcodeGetNextU8Jmp(pVCpu))
2009#endif /* IEM_WITH_SETJMP */
2010
2011
2012#ifndef IEM_WITH_SETJMP
2013/**
2014 * Fetches the next signed byte from the opcode stream.
2015 *
2016 * @returns Strict VBox status code.
2017 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2018 * @param pi8 Where to return the signed byte.
2019 */
2020DECLINLINE(VBOXSTRICTRC) iemOpcodeGetNextS8(PVMCPU pVCpu, int8_t *pi8)
2021{
2022 return iemOpcodeGetNextU8(pVCpu, (uint8_t *)pi8);
2023}
2024#endif /* !IEM_WITH_SETJMP */
2025
2026
2027/**
2028 * Fetches the next signed byte from the opcode stream, returning automatically
2029 * on failure.
2030 *
2031 * @param a_pi8 Where to return the signed byte.
2032 * @remark Implicitly references pVCpu.
2033 */
2034#ifndef IEM_WITH_SETJMP
2035# define IEM_OPCODE_GET_NEXT_S8(a_pi8) \
2036 do \
2037 { \
2038 VBOXSTRICTRC rcStrict2 = iemOpcodeGetNextS8(pVCpu, (a_pi8)); \
2039 if (rcStrict2 != VINF_SUCCESS) \
2040 return rcStrict2; \
2041 } while (0)
2042#else /* IEM_WITH_SETJMP */
2043# define IEM_OPCODE_GET_NEXT_S8(a_pi8) (*(a_pi8) = (int8_t)iemOpcodeGetNextU8Jmp(pVCpu))
2044
2045#endif /* IEM_WITH_SETJMP */
2046
2047#ifndef IEM_WITH_SETJMP
2048
2049/**
2050 * Deals with the problematic cases that iemOpcodeGetNextS8SxU16 doesn't like.
2051 *
2052 * @returns Strict VBox status code.
2053 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2054 * @param pu16 Where to return the opcode dword.
2055 */
2056DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemOpcodeGetNextS8SxU16Slow(PVMCPU pVCpu, uint16_t *pu16)
2057{
2058 uint8_t u8;
2059 VBOXSTRICTRC rcStrict = iemOpcodeGetNextU8Slow(pVCpu, &u8);
2060 if (rcStrict == VINF_SUCCESS)
2061 *pu16 = (int8_t)u8;
2062 return rcStrict;
2063}
2064
2065
2066/**
2067 * Fetches the next signed byte from the opcode stream, extending it to
2068 * unsigned 16-bit.
2069 *
2070 * @returns Strict VBox status code.
2071 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2072 * @param pu16 Where to return the unsigned word.
2073 */
2074DECLINLINE(VBOXSTRICTRC) iemOpcodeGetNextS8SxU16(PVMCPU pVCpu, uint16_t *pu16)
2075{
2076 uint8_t const offOpcode = pVCpu->iem.s.offOpcode;
2077 if (RT_UNLIKELY(offOpcode >= pVCpu->iem.s.cbOpcode))
2078 return iemOpcodeGetNextS8SxU16Slow(pVCpu, pu16);
2079
2080 *pu16 = (int8_t)pVCpu->iem.s.abOpcode[offOpcode];
2081 pVCpu->iem.s.offOpcode = offOpcode + 1;
2082 return VINF_SUCCESS;
2083}
2084
2085#endif /* !IEM_WITH_SETJMP */
2086
2087/**
2088 * Fetches the next signed byte from the opcode stream and sign-extending it to
2089 * a word, returning automatically on failure.
2090 *
2091 * @param a_pu16 Where to return the word.
2092 * @remark Implicitly references pVCpu.
2093 */
2094#ifndef IEM_WITH_SETJMP
2095# define IEM_OPCODE_GET_NEXT_S8_SX_U16(a_pu16) \
2096 do \
2097 { \
2098 VBOXSTRICTRC rcStrict2 = iemOpcodeGetNextS8SxU16(pVCpu, (a_pu16)); \
2099 if (rcStrict2 != VINF_SUCCESS) \
2100 return rcStrict2; \
2101 } while (0)
2102#else
2103# define IEM_OPCODE_GET_NEXT_S8_SX_U16(a_pu16) (*(a_pu16) = (int8_t)iemOpcodeGetNextU8Jmp(pVCpu))
2104#endif
2105
2106#ifndef IEM_WITH_SETJMP
2107
2108/**
2109 * Deals with the problematic cases that iemOpcodeGetNextS8SxU32 doesn't like.
2110 *
2111 * @returns Strict VBox status code.
2112 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2113 * @param pu32 Where to return the opcode dword.
2114 */
2115DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemOpcodeGetNextS8SxU32Slow(PVMCPU pVCpu, uint32_t *pu32)
2116{
2117 uint8_t u8;
2118 VBOXSTRICTRC rcStrict = iemOpcodeGetNextU8Slow(pVCpu, &u8);
2119 if (rcStrict == VINF_SUCCESS)
2120 *pu32 = (int8_t)u8;
2121 return rcStrict;
2122}
2123
2124
2125/**
2126 * Fetches the next signed byte from the opcode stream, extending it to
2127 * unsigned 32-bit.
2128 *
2129 * @returns Strict VBox status code.
2130 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2131 * @param pu32 Where to return the unsigned dword.
2132 */
2133DECLINLINE(VBOXSTRICTRC) iemOpcodeGetNextS8SxU32(PVMCPU pVCpu, uint32_t *pu32)
2134{
2135 uint8_t const offOpcode = pVCpu->iem.s.offOpcode;
2136 if (RT_UNLIKELY(offOpcode >= pVCpu->iem.s.cbOpcode))
2137 return iemOpcodeGetNextS8SxU32Slow(pVCpu, pu32);
2138
2139 *pu32 = (int8_t)pVCpu->iem.s.abOpcode[offOpcode];
2140 pVCpu->iem.s.offOpcode = offOpcode + 1;
2141 return VINF_SUCCESS;
2142}
2143
2144#endif /* !IEM_WITH_SETJMP */
2145
2146/**
2147 * Fetches the next signed byte from the opcode stream and sign-extending it to
2148 * a word, returning automatically on failure.
2149 *
2150 * @param a_pu32 Where to return the word.
2151 * @remark Implicitly references pVCpu.
2152 */
2153#ifndef IEM_WITH_SETJMP
2154#define IEM_OPCODE_GET_NEXT_S8_SX_U32(a_pu32) \
2155 do \
2156 { \
2157 VBOXSTRICTRC rcStrict2 = iemOpcodeGetNextS8SxU32(pVCpu, (a_pu32)); \
2158 if (rcStrict2 != VINF_SUCCESS) \
2159 return rcStrict2; \
2160 } while (0)
2161#else
2162# define IEM_OPCODE_GET_NEXT_S8_SX_U32(a_pu32) (*(a_pu32) = (int8_t)iemOpcodeGetNextU8Jmp(pVCpu))
2163#endif
2164
2165#ifndef IEM_WITH_SETJMP
2166
2167/**
2168 * Deals with the problematic cases that iemOpcodeGetNextS8SxU64 doesn't like.
2169 *
2170 * @returns Strict VBox status code.
2171 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2172 * @param pu64 Where to return the opcode qword.
2173 */
2174DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemOpcodeGetNextS8SxU64Slow(PVMCPU pVCpu, uint64_t *pu64)
2175{
2176 uint8_t u8;
2177 VBOXSTRICTRC rcStrict = iemOpcodeGetNextU8Slow(pVCpu, &u8);
2178 if (rcStrict == VINF_SUCCESS)
2179 *pu64 = (int8_t)u8;
2180 return rcStrict;
2181}
2182
2183
2184/**
2185 * Fetches the next signed byte from the opcode stream, extending it to
2186 * unsigned 64-bit.
2187 *
2188 * @returns Strict VBox status code.
2189 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2190 * @param pu64 Where to return the unsigned qword.
2191 */
2192DECLINLINE(VBOXSTRICTRC) iemOpcodeGetNextS8SxU64(PVMCPU pVCpu, uint64_t *pu64)
2193{
2194 uint8_t const offOpcode = pVCpu->iem.s.offOpcode;
2195 if (RT_UNLIKELY(offOpcode >= pVCpu->iem.s.cbOpcode))
2196 return iemOpcodeGetNextS8SxU64Slow(pVCpu, pu64);
2197
2198 *pu64 = (int8_t)pVCpu->iem.s.abOpcode[offOpcode];
2199 pVCpu->iem.s.offOpcode = offOpcode + 1;
2200 return VINF_SUCCESS;
2201}
2202
2203#endif /* !IEM_WITH_SETJMP */
2204
2205
2206/**
2207 * Fetches the next signed byte from the opcode stream and sign-extending it to
2208 * a word, returning automatically on failure.
2209 *
2210 * @param a_pu64 Where to return the word.
2211 * @remark Implicitly references pVCpu.
2212 */
2213#ifndef IEM_WITH_SETJMP
2214# define IEM_OPCODE_GET_NEXT_S8_SX_U64(a_pu64) \
2215 do \
2216 { \
2217 VBOXSTRICTRC rcStrict2 = iemOpcodeGetNextS8SxU64(pVCpu, (a_pu64)); \
2218 if (rcStrict2 != VINF_SUCCESS) \
2219 return rcStrict2; \
2220 } while (0)
2221#else
2222# define IEM_OPCODE_GET_NEXT_S8_SX_U64(a_pu64) (*(a_pu64) = (int8_t)iemOpcodeGetNextU8Jmp(pVCpu))
2223#endif
2224
2225
2226#ifndef IEM_WITH_SETJMP
2227
2228/**
2229 * Deals with the problematic cases that iemOpcodeGetNextU16 doesn't like.
2230 *
2231 * @returns Strict VBox status code.
2232 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2233 * @param pu16 Where to return the opcode word.
2234 */
2235DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemOpcodeGetNextU16Slow(PVMCPU pVCpu, uint16_t *pu16)
2236{
2237 VBOXSTRICTRC rcStrict = iemOpcodeFetchMoreBytes(pVCpu, 2);
2238 if (rcStrict == VINF_SUCCESS)
2239 {
2240 uint8_t offOpcode = pVCpu->iem.s.offOpcode;
2241# ifdef IEM_USE_UNALIGNED_DATA_ACCESS
2242 *pu16 = *(uint16_t const *)&pVCpu->iem.s.abOpcode[offOpcode];
2243# else
2244 *pu16 = RT_MAKE_U16(pVCpu->iem.s.abOpcode[offOpcode], pVCpu->iem.s.abOpcode[offOpcode + 1]);
2245# endif
2246 pVCpu->iem.s.offOpcode = offOpcode + 2;
2247 }
2248 else
2249 *pu16 = 0;
2250 return rcStrict;
2251}
2252
2253
2254/**
2255 * Fetches the next opcode word.
2256 *
2257 * @returns Strict VBox status code.
2258 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2259 * @param pu16 Where to return the opcode word.
2260 */
2261DECLINLINE(VBOXSTRICTRC) iemOpcodeGetNextU16(PVMCPU pVCpu, uint16_t *pu16)
2262{
2263 uintptr_t const offOpcode = pVCpu->iem.s.offOpcode;
2264 if (RT_LIKELY((uint8_t)offOpcode + 2 <= pVCpu->iem.s.cbOpcode))
2265 {
2266 pVCpu->iem.s.offOpcode = (uint8_t)offOpcode + 2;
2267# ifdef IEM_USE_UNALIGNED_DATA_ACCESS
2268 *pu16 = *(uint16_t const *)&pVCpu->iem.s.abOpcode[offOpcode];
2269# else
2270 *pu16 = RT_MAKE_U16(pVCpu->iem.s.abOpcode[offOpcode], pVCpu->iem.s.abOpcode[offOpcode + 1]);
2271# endif
2272 return VINF_SUCCESS;
2273 }
2274 return iemOpcodeGetNextU16Slow(pVCpu, pu16);
2275}
2276
2277#else /* IEM_WITH_SETJMP */
2278
2279/**
2280 * Deals with the problematic cases that iemOpcodeGetNextU16Jmp doesn't like, longjmp on error
2281 *
2282 * @returns The opcode word.
2283 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2284 */
2285DECL_NO_INLINE(IEM_STATIC, uint16_t) iemOpcodeGetNextU16SlowJmp(PVMCPU pVCpu)
2286{
2287# ifdef IEM_WITH_CODE_TLB
2288 uint16_t u16;
2289 iemOpcodeFetchBytesJmp(pVCpu, sizeof(u16), &u16);
2290 return u16;
2291# else
2292 VBOXSTRICTRC rcStrict = iemOpcodeFetchMoreBytes(pVCpu, 2);
2293 if (rcStrict == VINF_SUCCESS)
2294 {
2295 uint8_t offOpcode = pVCpu->iem.s.offOpcode;
2296 pVCpu->iem.s.offOpcode += 2;
2297# ifdef IEM_USE_UNALIGNED_DATA_ACCESS
2298 return *(uint16_t const *)&pVCpu->iem.s.abOpcode[offOpcode];
2299# else
2300 return RT_MAKE_U16(pVCpu->iem.s.abOpcode[offOpcode], pVCpu->iem.s.abOpcode[offOpcode + 1]);
2301# endif
2302 }
2303 longjmp(*pVCpu->iem.s.CTX_SUFF(pJmpBuf), VBOXSTRICTRC_VAL(rcStrict));
2304# endif
2305}
2306
2307
2308/**
2309 * Fetches the next opcode word, longjmp on error.
2310 *
2311 * @returns The opcode word.
2312 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2313 */
2314DECLINLINE(uint16_t) iemOpcodeGetNextU16Jmp(PVMCPU pVCpu)
2315{
2316# ifdef IEM_WITH_CODE_TLB
2317 uintptr_t offBuf = pVCpu->iem.s.offInstrNextByte;
2318 uint8_t const *pbBuf = pVCpu->iem.s.pbInstrBuf;
2319 if (RT_LIKELY( pbBuf != NULL
2320 && offBuf + 2 <= pVCpu->iem.s.cbInstrBuf))
2321 {
2322 pVCpu->iem.s.offInstrNextByte = (uint32_t)offBuf + 2;
2323# ifdef IEM_USE_UNALIGNED_DATA_ACCESS
2324 return *(uint16_t const *)&pbBuf[offBuf];
2325# else
2326 return RT_MAKE_U16(pbBuf[offBuf], pbBuf[offBuf + 1]);
2327# endif
2328 }
2329# else
2330 uintptr_t const offOpcode = pVCpu->iem.s.offOpcode;
2331 if (RT_LIKELY((uint8_t)offOpcode + 2 <= pVCpu->iem.s.cbOpcode))
2332 {
2333 pVCpu->iem.s.offOpcode = (uint8_t)offOpcode + 2;
2334# ifdef IEM_USE_UNALIGNED_DATA_ACCESS
2335 return *(uint16_t const *)&pVCpu->iem.s.abOpcode[offOpcode];
2336# else
2337 return RT_MAKE_U16(pVCpu->iem.s.abOpcode[offOpcode], pVCpu->iem.s.abOpcode[offOpcode + 1]);
2338# endif
2339 }
2340# endif
2341 return iemOpcodeGetNextU16SlowJmp(pVCpu);
2342}
2343
2344#endif /* IEM_WITH_SETJMP */
2345
2346
2347/**
2348 * Fetches the next opcode word, returns automatically on failure.
2349 *
2350 * @param a_pu16 Where to return the opcode word.
2351 * @remark Implicitly references pVCpu.
2352 */
2353#ifndef IEM_WITH_SETJMP
2354# define IEM_OPCODE_GET_NEXT_U16(a_pu16) \
2355 do \
2356 { \
2357 VBOXSTRICTRC rcStrict2 = iemOpcodeGetNextU16(pVCpu, (a_pu16)); \
2358 if (rcStrict2 != VINF_SUCCESS) \
2359 return rcStrict2; \
2360 } while (0)
2361#else
2362# define IEM_OPCODE_GET_NEXT_U16(a_pu16) (*(a_pu16) = iemOpcodeGetNextU16Jmp(pVCpu))
2363#endif
2364
2365#ifndef IEM_WITH_SETJMP
2366
2367/**
2368 * Deals with the problematic cases that iemOpcodeGetNextU16ZxU32 doesn't like.
2369 *
2370 * @returns Strict VBox status code.
2371 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2372 * @param pu32 Where to return the opcode double word.
2373 */
2374DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemOpcodeGetNextU16ZxU32Slow(PVMCPU pVCpu, uint32_t *pu32)
2375{
2376 VBOXSTRICTRC rcStrict = iemOpcodeFetchMoreBytes(pVCpu, 2);
2377 if (rcStrict == VINF_SUCCESS)
2378 {
2379 uint8_t offOpcode = pVCpu->iem.s.offOpcode;
2380 *pu32 = RT_MAKE_U16(pVCpu->iem.s.abOpcode[offOpcode], pVCpu->iem.s.abOpcode[offOpcode + 1]);
2381 pVCpu->iem.s.offOpcode = offOpcode + 2;
2382 }
2383 else
2384 *pu32 = 0;
2385 return rcStrict;
2386}
2387
2388
2389/**
2390 * Fetches the next opcode word, zero extending it to a double word.
2391 *
2392 * @returns Strict VBox status code.
2393 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2394 * @param pu32 Where to return the opcode double word.
2395 */
2396DECLINLINE(VBOXSTRICTRC) iemOpcodeGetNextU16ZxU32(PVMCPU pVCpu, uint32_t *pu32)
2397{
2398 uint8_t const offOpcode = pVCpu->iem.s.offOpcode;
2399 if (RT_UNLIKELY(offOpcode + 2 > pVCpu->iem.s.cbOpcode))
2400 return iemOpcodeGetNextU16ZxU32Slow(pVCpu, pu32);
2401
2402 *pu32 = RT_MAKE_U16(pVCpu->iem.s.abOpcode[offOpcode], pVCpu->iem.s.abOpcode[offOpcode + 1]);
2403 pVCpu->iem.s.offOpcode = offOpcode + 2;
2404 return VINF_SUCCESS;
2405}
2406
2407#endif /* !IEM_WITH_SETJMP */
2408
2409
2410/**
2411 * Fetches the next opcode word and zero extends it to a double word, returns
2412 * automatically on failure.
2413 *
2414 * @param a_pu32 Where to return the opcode double word.
2415 * @remark Implicitly references pVCpu.
2416 */
2417#ifndef IEM_WITH_SETJMP
2418# define IEM_OPCODE_GET_NEXT_U16_ZX_U32(a_pu32) \
2419 do \
2420 { \
2421 VBOXSTRICTRC rcStrict2 = iemOpcodeGetNextU16ZxU32(pVCpu, (a_pu32)); \
2422 if (rcStrict2 != VINF_SUCCESS) \
2423 return rcStrict2; \
2424 } while (0)
2425#else
2426# define IEM_OPCODE_GET_NEXT_U16_ZX_U32(a_pu32) (*(a_pu32) = iemOpcodeGetNextU16Jmp(pVCpu))
2427#endif
2428
2429#ifndef IEM_WITH_SETJMP
2430
2431/**
2432 * Deals with the problematic cases that iemOpcodeGetNextU16ZxU64 doesn't like.
2433 *
2434 * @returns Strict VBox status code.
2435 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2436 * @param pu64 Where to return the opcode quad word.
2437 */
2438DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemOpcodeGetNextU16ZxU64Slow(PVMCPU pVCpu, uint64_t *pu64)
2439{
2440 VBOXSTRICTRC rcStrict = iemOpcodeFetchMoreBytes(pVCpu, 2);
2441 if (rcStrict == VINF_SUCCESS)
2442 {
2443 uint8_t offOpcode = pVCpu->iem.s.offOpcode;
2444 *pu64 = RT_MAKE_U16(pVCpu->iem.s.abOpcode[offOpcode], pVCpu->iem.s.abOpcode[offOpcode + 1]);
2445 pVCpu->iem.s.offOpcode = offOpcode + 2;
2446 }
2447 else
2448 *pu64 = 0;
2449 return rcStrict;
2450}
2451
2452
2453/**
2454 * Fetches the next opcode word, zero extending it to a quad word.
2455 *
2456 * @returns Strict VBox status code.
2457 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2458 * @param pu64 Where to return the opcode quad word.
2459 */
2460DECLINLINE(VBOXSTRICTRC) iemOpcodeGetNextU16ZxU64(PVMCPU pVCpu, uint64_t *pu64)
2461{
2462 uint8_t const offOpcode = pVCpu->iem.s.offOpcode;
2463 if (RT_UNLIKELY(offOpcode + 2 > pVCpu->iem.s.cbOpcode))
2464 return iemOpcodeGetNextU16ZxU64Slow(pVCpu, pu64);
2465
2466 *pu64 = RT_MAKE_U16(pVCpu->iem.s.abOpcode[offOpcode], pVCpu->iem.s.abOpcode[offOpcode + 1]);
2467 pVCpu->iem.s.offOpcode = offOpcode + 2;
2468 return VINF_SUCCESS;
2469}
2470
2471#endif /* !IEM_WITH_SETJMP */
2472
2473/**
2474 * Fetches the next opcode word and zero extends it to a quad word, returns
2475 * automatically on failure.
2476 *
2477 * @param a_pu64 Where to return the opcode quad word.
2478 * @remark Implicitly references pVCpu.
2479 */
2480#ifndef IEM_WITH_SETJMP
2481# define IEM_OPCODE_GET_NEXT_U16_ZX_U64(a_pu64) \
2482 do \
2483 { \
2484 VBOXSTRICTRC rcStrict2 = iemOpcodeGetNextU16ZxU64(pVCpu, (a_pu64)); \
2485 if (rcStrict2 != VINF_SUCCESS) \
2486 return rcStrict2; \
2487 } while (0)
2488#else
2489# define IEM_OPCODE_GET_NEXT_U16_ZX_U64(a_pu64) (*(a_pu64) = iemOpcodeGetNextU16Jmp(pVCpu))
2490#endif
2491
2492
2493#ifndef IEM_WITH_SETJMP
2494/**
2495 * Fetches the next signed word from the opcode stream.
2496 *
2497 * @returns Strict VBox status code.
2498 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2499 * @param pi16 Where to return the signed word.
2500 */
2501DECLINLINE(VBOXSTRICTRC) iemOpcodeGetNextS16(PVMCPU pVCpu, int16_t *pi16)
2502{
2503 return iemOpcodeGetNextU16(pVCpu, (uint16_t *)pi16);
2504}
2505#endif /* !IEM_WITH_SETJMP */
2506
2507
2508/**
2509 * Fetches the next signed word from the opcode stream, returning automatically
2510 * on failure.
2511 *
2512 * @param a_pi16 Where to return the signed word.
2513 * @remark Implicitly references pVCpu.
2514 */
2515#ifndef IEM_WITH_SETJMP
2516# define IEM_OPCODE_GET_NEXT_S16(a_pi16) \
2517 do \
2518 { \
2519 VBOXSTRICTRC rcStrict2 = iemOpcodeGetNextS16(pVCpu, (a_pi16)); \
2520 if (rcStrict2 != VINF_SUCCESS) \
2521 return rcStrict2; \
2522 } while (0)
2523#else
2524# define IEM_OPCODE_GET_NEXT_S16(a_pi16) (*(a_pi16) = (int16_t)iemOpcodeGetNextU16Jmp(pVCpu))
2525#endif
2526
2527#ifndef IEM_WITH_SETJMP
2528
2529/**
2530 * Deals with the problematic cases that iemOpcodeGetNextU32 doesn't like.
2531 *
2532 * @returns Strict VBox status code.
2533 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2534 * @param pu32 Where to return the opcode dword.
2535 */
2536DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemOpcodeGetNextU32Slow(PVMCPU pVCpu, uint32_t *pu32)
2537{
2538 VBOXSTRICTRC rcStrict = iemOpcodeFetchMoreBytes(pVCpu, 4);
2539 if (rcStrict == VINF_SUCCESS)
2540 {
2541 uint8_t offOpcode = pVCpu->iem.s.offOpcode;
2542# ifdef IEM_USE_UNALIGNED_DATA_ACCESS
2543 *pu32 = *(uint32_t const *)&pVCpu->iem.s.abOpcode[offOpcode];
2544# else
2545 *pu32 = RT_MAKE_U32_FROM_U8(pVCpu->iem.s.abOpcode[offOpcode],
2546 pVCpu->iem.s.abOpcode[offOpcode + 1],
2547 pVCpu->iem.s.abOpcode[offOpcode + 2],
2548 pVCpu->iem.s.abOpcode[offOpcode + 3]);
2549# endif
2550 pVCpu->iem.s.offOpcode = offOpcode + 4;
2551 }
2552 else
2553 *pu32 = 0;
2554 return rcStrict;
2555}
2556
2557
2558/**
2559 * Fetches the next opcode dword.
2560 *
2561 * @returns Strict VBox status code.
2562 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2563 * @param pu32 Where to return the opcode double word.
2564 */
2565DECLINLINE(VBOXSTRICTRC) iemOpcodeGetNextU32(PVMCPU pVCpu, uint32_t *pu32)
2566{
2567 uintptr_t const offOpcode = pVCpu->iem.s.offOpcode;
2568 if (RT_LIKELY((uint8_t)offOpcode + 4 <= pVCpu->iem.s.cbOpcode))
2569 {
2570 pVCpu->iem.s.offOpcode = (uint8_t)offOpcode + 4;
2571# ifdef IEM_USE_UNALIGNED_DATA_ACCESS
2572 *pu32 = *(uint32_t const *)&pVCpu->iem.s.abOpcode[offOpcode];
2573# else
2574 *pu32 = RT_MAKE_U32_FROM_U8(pVCpu->iem.s.abOpcode[offOpcode],
2575 pVCpu->iem.s.abOpcode[offOpcode + 1],
2576 pVCpu->iem.s.abOpcode[offOpcode + 2],
2577 pVCpu->iem.s.abOpcode[offOpcode + 3]);
2578# endif
2579 return VINF_SUCCESS;
2580 }
2581 return iemOpcodeGetNextU32Slow(pVCpu, pu32);
2582}
2583
2584#else /* !IEM_WITH_SETJMP */
2585
2586/**
2587 * Deals with the problematic cases that iemOpcodeGetNextU32Jmp doesn't like, longjmp on error.
2588 *
2589 * @returns The opcode dword.
2590 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2591 */
2592DECL_NO_INLINE(IEM_STATIC, uint32_t) iemOpcodeGetNextU32SlowJmp(PVMCPU pVCpu)
2593{
2594# ifdef IEM_WITH_CODE_TLB
2595 uint32_t u32;
2596 iemOpcodeFetchBytesJmp(pVCpu, sizeof(u32), &u32);
2597 return u32;
2598# else
2599 VBOXSTRICTRC rcStrict = iemOpcodeFetchMoreBytes(pVCpu, 4);
2600 if (rcStrict == VINF_SUCCESS)
2601 {
2602 uint8_t offOpcode = pVCpu->iem.s.offOpcode;
2603 pVCpu->iem.s.offOpcode = offOpcode + 4;
2604# ifdef IEM_USE_UNALIGNED_DATA_ACCESS
2605 return *(uint32_t const *)&pVCpu->iem.s.abOpcode[offOpcode];
2606# else
2607 return RT_MAKE_U32_FROM_U8(pVCpu->iem.s.abOpcode[offOpcode],
2608 pVCpu->iem.s.abOpcode[offOpcode + 1],
2609 pVCpu->iem.s.abOpcode[offOpcode + 2],
2610 pVCpu->iem.s.abOpcode[offOpcode + 3]);
2611# endif
2612 }
2613 longjmp(*pVCpu->iem.s.CTX_SUFF(pJmpBuf), VBOXSTRICTRC_VAL(rcStrict));
2614# endif
2615}
2616
2617
2618/**
2619 * Fetches the next opcode dword, longjmp on error.
2620 *
2621 * @returns The opcode dword.
2622 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2623 */
2624DECLINLINE(uint32_t) iemOpcodeGetNextU32Jmp(PVMCPU pVCpu)
2625{
2626# ifdef IEM_WITH_CODE_TLB
2627 uintptr_t offBuf = pVCpu->iem.s.offInstrNextByte;
2628 uint8_t const *pbBuf = pVCpu->iem.s.pbInstrBuf;
2629 if (RT_LIKELY( pbBuf != NULL
2630 && offBuf + 4 <= pVCpu->iem.s.cbInstrBuf))
2631 {
2632 pVCpu->iem.s.offInstrNextByte = (uint32_t)offBuf + 4;
2633# ifdef IEM_USE_UNALIGNED_DATA_ACCESS
2634 return *(uint32_t const *)&pbBuf[offBuf];
2635# else
2636 return RT_MAKE_U32_FROM_U8(pbBuf[offBuf],
2637 pbBuf[offBuf + 1],
2638 pbBuf[offBuf + 2],
2639 pbBuf[offBuf + 3]);
2640# endif
2641 }
2642# else
2643 uintptr_t const offOpcode = pVCpu->iem.s.offOpcode;
2644 if (RT_LIKELY((uint8_t)offOpcode + 4 <= pVCpu->iem.s.cbOpcode))
2645 {
2646 pVCpu->iem.s.offOpcode = (uint8_t)offOpcode + 4;
2647# ifdef IEM_USE_UNALIGNED_DATA_ACCESS
2648 return *(uint32_t const *)&pVCpu->iem.s.abOpcode[offOpcode];
2649# else
2650 return RT_MAKE_U32_FROM_U8(pVCpu->iem.s.abOpcode[offOpcode],
2651 pVCpu->iem.s.abOpcode[offOpcode + 1],
2652 pVCpu->iem.s.abOpcode[offOpcode + 2],
2653 pVCpu->iem.s.abOpcode[offOpcode + 3]);
2654# endif
2655 }
2656# endif
2657 return iemOpcodeGetNextU32SlowJmp(pVCpu);
2658}
2659
2660#endif /* !IEM_WITH_SETJMP */
2661
2662
2663/**
2664 * Fetches the next opcode dword, returns automatically on failure.
2665 *
2666 * @param a_pu32 Where to return the opcode dword.
2667 * @remark Implicitly references pVCpu.
2668 */
2669#ifndef IEM_WITH_SETJMP
2670# define IEM_OPCODE_GET_NEXT_U32(a_pu32) \
2671 do \
2672 { \
2673 VBOXSTRICTRC rcStrict2 = iemOpcodeGetNextU32(pVCpu, (a_pu32)); \
2674 if (rcStrict2 != VINF_SUCCESS) \
2675 return rcStrict2; \
2676 } while (0)
2677#else
2678# define IEM_OPCODE_GET_NEXT_U32(a_pu32) (*(a_pu32) = iemOpcodeGetNextU32Jmp(pVCpu))
2679#endif
2680
2681#ifndef IEM_WITH_SETJMP
2682
2683/**
2684 * Deals with the problematic cases that iemOpcodeGetNextU32ZxU64 doesn't like.
2685 *
2686 * @returns Strict VBox status code.
2687 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2688 * @param pu64 Where to return the opcode dword.
2689 */
2690DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemOpcodeGetNextU32ZxU64Slow(PVMCPU pVCpu, uint64_t *pu64)
2691{
2692 VBOXSTRICTRC rcStrict = iemOpcodeFetchMoreBytes(pVCpu, 4);
2693 if (rcStrict == VINF_SUCCESS)
2694 {
2695 uint8_t offOpcode = pVCpu->iem.s.offOpcode;
2696 *pu64 = RT_MAKE_U32_FROM_U8(pVCpu->iem.s.abOpcode[offOpcode],
2697 pVCpu->iem.s.abOpcode[offOpcode + 1],
2698 pVCpu->iem.s.abOpcode[offOpcode + 2],
2699 pVCpu->iem.s.abOpcode[offOpcode + 3]);
2700 pVCpu->iem.s.offOpcode = offOpcode + 4;
2701 }
2702 else
2703 *pu64 = 0;
2704 return rcStrict;
2705}
2706
2707
2708/**
2709 * Fetches the next opcode dword, zero extending it to a quad word.
2710 *
2711 * @returns Strict VBox status code.
2712 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2713 * @param pu64 Where to return the opcode quad word.
2714 */
2715DECLINLINE(VBOXSTRICTRC) iemOpcodeGetNextU32ZxU64(PVMCPU pVCpu, uint64_t *pu64)
2716{
2717 uint8_t const offOpcode = pVCpu->iem.s.offOpcode;
2718 if (RT_UNLIKELY(offOpcode + 4 > pVCpu->iem.s.cbOpcode))
2719 return iemOpcodeGetNextU32ZxU64Slow(pVCpu, pu64);
2720
2721 *pu64 = RT_MAKE_U32_FROM_U8(pVCpu->iem.s.abOpcode[offOpcode],
2722 pVCpu->iem.s.abOpcode[offOpcode + 1],
2723 pVCpu->iem.s.abOpcode[offOpcode + 2],
2724 pVCpu->iem.s.abOpcode[offOpcode + 3]);
2725 pVCpu->iem.s.offOpcode = offOpcode + 4;
2726 return VINF_SUCCESS;
2727}
2728
2729#endif /* !IEM_WITH_SETJMP */
2730
2731
2732/**
2733 * Fetches the next opcode dword and zero extends it to a quad word, returns
2734 * automatically on failure.
2735 *
2736 * @param a_pu64 Where to return the opcode quad word.
2737 * @remark Implicitly references pVCpu.
2738 */
2739#ifndef IEM_WITH_SETJMP
2740# define IEM_OPCODE_GET_NEXT_U32_ZX_U64(a_pu64) \
2741 do \
2742 { \
2743 VBOXSTRICTRC rcStrict2 = iemOpcodeGetNextU32ZxU64(pVCpu, (a_pu64)); \
2744 if (rcStrict2 != VINF_SUCCESS) \
2745 return rcStrict2; \
2746 } while (0)
2747#else
2748# define IEM_OPCODE_GET_NEXT_U32_ZX_U64(a_pu64) (*(a_pu64) = iemOpcodeGetNextU32Jmp(pVCpu))
2749#endif
2750
2751
2752#ifndef IEM_WITH_SETJMP
2753/**
2754 * Fetches the next signed double word from the opcode stream.
2755 *
2756 * @returns Strict VBox status code.
2757 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2758 * @param pi32 Where to return the signed double word.
2759 */
2760DECLINLINE(VBOXSTRICTRC) iemOpcodeGetNextS32(PVMCPU pVCpu, int32_t *pi32)
2761{
2762 return iemOpcodeGetNextU32(pVCpu, (uint32_t *)pi32);
2763}
2764#endif
2765
2766/**
2767 * Fetches the next signed double word from the opcode stream, returning
2768 * automatically on failure.
2769 *
2770 * @param a_pi32 Where to return the signed double word.
2771 * @remark Implicitly references pVCpu.
2772 */
2773#ifndef IEM_WITH_SETJMP
2774# define IEM_OPCODE_GET_NEXT_S32(a_pi32) \
2775 do \
2776 { \
2777 VBOXSTRICTRC rcStrict2 = iemOpcodeGetNextS32(pVCpu, (a_pi32)); \
2778 if (rcStrict2 != VINF_SUCCESS) \
2779 return rcStrict2; \
2780 } while (0)
2781#else
2782# define IEM_OPCODE_GET_NEXT_S32(a_pi32) (*(a_pi32) = (int32_t)iemOpcodeGetNextU32Jmp(pVCpu))
2783#endif
2784
2785#ifndef IEM_WITH_SETJMP
2786
2787/**
2788 * Deals with the problematic cases that iemOpcodeGetNextS32SxU64 doesn't like.
2789 *
2790 * @returns Strict VBox status code.
2791 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2792 * @param pu64 Where to return the opcode qword.
2793 */
2794DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemOpcodeGetNextS32SxU64Slow(PVMCPU pVCpu, uint64_t *pu64)
2795{
2796 VBOXSTRICTRC rcStrict = iemOpcodeFetchMoreBytes(pVCpu, 4);
2797 if (rcStrict == VINF_SUCCESS)
2798 {
2799 uint8_t offOpcode = pVCpu->iem.s.offOpcode;
2800 *pu64 = (int32_t)RT_MAKE_U32_FROM_U8(pVCpu->iem.s.abOpcode[offOpcode],
2801 pVCpu->iem.s.abOpcode[offOpcode + 1],
2802 pVCpu->iem.s.abOpcode[offOpcode + 2],
2803 pVCpu->iem.s.abOpcode[offOpcode + 3]);
2804 pVCpu->iem.s.offOpcode = offOpcode + 4;
2805 }
2806 else
2807 *pu64 = 0;
2808 return rcStrict;
2809}
2810
2811
2812/**
2813 * Fetches the next opcode dword, sign extending it into a quad word.
2814 *
2815 * @returns Strict VBox status code.
2816 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2817 * @param pu64 Where to return the opcode quad word.
2818 */
2819DECLINLINE(VBOXSTRICTRC) iemOpcodeGetNextS32SxU64(PVMCPU pVCpu, uint64_t *pu64)
2820{
2821 uint8_t const offOpcode = pVCpu->iem.s.offOpcode;
2822 if (RT_UNLIKELY(offOpcode + 4 > pVCpu->iem.s.cbOpcode))
2823 return iemOpcodeGetNextS32SxU64Slow(pVCpu, pu64);
2824
2825 int32_t i32 = RT_MAKE_U32_FROM_U8(pVCpu->iem.s.abOpcode[offOpcode],
2826 pVCpu->iem.s.abOpcode[offOpcode + 1],
2827 pVCpu->iem.s.abOpcode[offOpcode + 2],
2828 pVCpu->iem.s.abOpcode[offOpcode + 3]);
2829 *pu64 = i32;
2830 pVCpu->iem.s.offOpcode = offOpcode + 4;
2831 return VINF_SUCCESS;
2832}
2833
2834#endif /* !IEM_WITH_SETJMP */
2835
2836
2837/**
2838 * Fetches the next opcode double word and sign extends it to a quad word,
2839 * returns automatically on failure.
2840 *
2841 * @param a_pu64 Where to return the opcode quad word.
2842 * @remark Implicitly references pVCpu.
2843 */
2844#ifndef IEM_WITH_SETJMP
2845# define IEM_OPCODE_GET_NEXT_S32_SX_U64(a_pu64) \
2846 do \
2847 { \
2848 VBOXSTRICTRC rcStrict2 = iemOpcodeGetNextS32SxU64(pVCpu, (a_pu64)); \
2849 if (rcStrict2 != VINF_SUCCESS) \
2850 return rcStrict2; \
2851 } while (0)
2852#else
2853# define IEM_OPCODE_GET_NEXT_S32_SX_U64(a_pu64) (*(a_pu64) = (int32_t)iemOpcodeGetNextU32Jmp(pVCpu))
2854#endif
2855
2856#ifndef IEM_WITH_SETJMP
2857
2858/**
2859 * Deals with the problematic cases that iemOpcodeGetNextU64 doesn't like.
2860 *
2861 * @returns Strict VBox status code.
2862 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2863 * @param pu64 Where to return the opcode qword.
2864 */
2865DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemOpcodeGetNextU64Slow(PVMCPU pVCpu, uint64_t *pu64)
2866{
2867 VBOXSTRICTRC rcStrict = iemOpcodeFetchMoreBytes(pVCpu, 8);
2868 if (rcStrict == VINF_SUCCESS)
2869 {
2870 uint8_t offOpcode = pVCpu->iem.s.offOpcode;
2871# ifdef IEM_USE_UNALIGNED_DATA_ACCESS
2872 *pu64 = *(uint64_t const *)&pVCpu->iem.s.abOpcode[offOpcode];
2873# else
2874 *pu64 = RT_MAKE_U64_FROM_U8(pVCpu->iem.s.abOpcode[offOpcode],
2875 pVCpu->iem.s.abOpcode[offOpcode + 1],
2876 pVCpu->iem.s.abOpcode[offOpcode + 2],
2877 pVCpu->iem.s.abOpcode[offOpcode + 3],
2878 pVCpu->iem.s.abOpcode[offOpcode + 4],
2879 pVCpu->iem.s.abOpcode[offOpcode + 5],
2880 pVCpu->iem.s.abOpcode[offOpcode + 6],
2881 pVCpu->iem.s.abOpcode[offOpcode + 7]);
2882# endif
2883 pVCpu->iem.s.offOpcode = offOpcode + 8;
2884 }
2885 else
2886 *pu64 = 0;
2887 return rcStrict;
2888}
2889
2890
2891/**
2892 * Fetches the next opcode qword.
2893 *
2894 * @returns Strict VBox status code.
2895 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2896 * @param pu64 Where to return the opcode qword.
2897 */
2898DECLINLINE(VBOXSTRICTRC) iemOpcodeGetNextU64(PVMCPU pVCpu, uint64_t *pu64)
2899{
2900 uintptr_t const offOpcode = pVCpu->iem.s.offOpcode;
2901 if (RT_LIKELY((uint8_t)offOpcode + 8 <= pVCpu->iem.s.cbOpcode))
2902 {
2903# ifdef IEM_USE_UNALIGNED_DATA_ACCESS
2904 *pu64 = *(uint64_t const *)&pVCpu->iem.s.abOpcode[offOpcode];
2905# else
2906 *pu64 = RT_MAKE_U64_FROM_U8(pVCpu->iem.s.abOpcode[offOpcode],
2907 pVCpu->iem.s.abOpcode[offOpcode + 1],
2908 pVCpu->iem.s.abOpcode[offOpcode + 2],
2909 pVCpu->iem.s.abOpcode[offOpcode + 3],
2910 pVCpu->iem.s.abOpcode[offOpcode + 4],
2911 pVCpu->iem.s.abOpcode[offOpcode + 5],
2912 pVCpu->iem.s.abOpcode[offOpcode + 6],
2913 pVCpu->iem.s.abOpcode[offOpcode + 7]);
2914# endif
2915 pVCpu->iem.s.offOpcode = (uint8_t)offOpcode + 8;
2916 return VINF_SUCCESS;
2917 }
2918 return iemOpcodeGetNextU64Slow(pVCpu, pu64);
2919}
2920
2921#else /* IEM_WITH_SETJMP */
2922
2923/**
2924 * Deals with the problematic cases that iemOpcodeGetNextU64Jmp doesn't like, longjmp on error.
2925 *
2926 * @returns The opcode qword.
2927 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2928 */
2929DECL_NO_INLINE(IEM_STATIC, uint64_t) iemOpcodeGetNextU64SlowJmp(PVMCPU pVCpu)
2930{
2931# ifdef IEM_WITH_CODE_TLB
2932 uint64_t u64;
2933 iemOpcodeFetchBytesJmp(pVCpu, sizeof(u64), &u64);
2934 return u64;
2935# else
2936 VBOXSTRICTRC rcStrict = iemOpcodeFetchMoreBytes(pVCpu, 8);
2937 if (rcStrict == VINF_SUCCESS)
2938 {
2939 uint8_t offOpcode = pVCpu->iem.s.offOpcode;
2940 pVCpu->iem.s.offOpcode = offOpcode + 8;
2941# ifdef IEM_USE_UNALIGNED_DATA_ACCESS
2942 return *(uint64_t const *)&pVCpu->iem.s.abOpcode[offOpcode];
2943# else
2944 return RT_MAKE_U64_FROM_U8(pVCpu->iem.s.abOpcode[offOpcode],
2945 pVCpu->iem.s.abOpcode[offOpcode + 1],
2946 pVCpu->iem.s.abOpcode[offOpcode + 2],
2947 pVCpu->iem.s.abOpcode[offOpcode + 3],
2948 pVCpu->iem.s.abOpcode[offOpcode + 4],
2949 pVCpu->iem.s.abOpcode[offOpcode + 5],
2950 pVCpu->iem.s.abOpcode[offOpcode + 6],
2951 pVCpu->iem.s.abOpcode[offOpcode + 7]);
2952# endif
2953 }
2954 longjmp(*pVCpu->iem.s.CTX_SUFF(pJmpBuf), VBOXSTRICTRC_VAL(rcStrict));
2955# endif
2956}
2957
2958
2959/**
2960 * Fetches the next opcode qword, longjmp on error.
2961 *
2962 * @returns The opcode qword.
2963 * @param pVCpu The cross context virtual CPU structure of the calling thread.
2964 */
2965DECLINLINE(uint64_t) iemOpcodeGetNextU64Jmp(PVMCPU pVCpu)
2966{
2967# ifdef IEM_WITH_CODE_TLB
2968 uintptr_t offBuf = pVCpu->iem.s.offInstrNextByte;
2969 uint8_t const *pbBuf = pVCpu->iem.s.pbInstrBuf;
2970 if (RT_LIKELY( pbBuf != NULL
2971 && offBuf + 8 <= pVCpu->iem.s.cbInstrBuf))
2972 {
2973 pVCpu->iem.s.offInstrNextByte = (uint32_t)offBuf + 8;
2974# ifdef IEM_USE_UNALIGNED_DATA_ACCESS
2975 return *(uint64_t const *)&pbBuf[offBuf];
2976# else
2977 return RT_MAKE_U64_FROM_U8(pbBuf[offBuf],
2978 pbBuf[offBuf + 1],
2979 pbBuf[offBuf + 2],
2980 pbBuf[offBuf + 3],
2981 pbBuf[offBuf + 4],
2982 pbBuf[offBuf + 5],
2983 pbBuf[offBuf + 6],
2984 pbBuf[offBuf + 7]);
2985# endif
2986 }
2987# else
2988 uintptr_t const offOpcode = pVCpu->iem.s.offOpcode;
2989 if (RT_LIKELY((uint8_t)offOpcode + 8 <= pVCpu->iem.s.cbOpcode))
2990 {
2991 pVCpu->iem.s.offOpcode = (uint8_t)offOpcode + 8;
2992# ifdef IEM_USE_UNALIGNED_DATA_ACCESS
2993 return *(uint64_t const *)&pVCpu->iem.s.abOpcode[offOpcode];
2994# else
2995 return RT_MAKE_U64_FROM_U8(pVCpu->iem.s.abOpcode[offOpcode],
2996 pVCpu->iem.s.abOpcode[offOpcode + 1],
2997 pVCpu->iem.s.abOpcode[offOpcode + 2],
2998 pVCpu->iem.s.abOpcode[offOpcode + 3],
2999 pVCpu->iem.s.abOpcode[offOpcode + 4],
3000 pVCpu->iem.s.abOpcode[offOpcode + 5],
3001 pVCpu->iem.s.abOpcode[offOpcode + 6],
3002 pVCpu->iem.s.abOpcode[offOpcode + 7]);
3003# endif
3004 }
3005# endif
3006 return iemOpcodeGetNextU64SlowJmp(pVCpu);
3007}
3008
3009#endif /* IEM_WITH_SETJMP */
3010
3011/**
3012 * Fetches the next opcode quad word, returns automatically on failure.
3013 *
3014 * @param a_pu64 Where to return the opcode quad word.
3015 * @remark Implicitly references pVCpu.
3016 */
3017#ifndef IEM_WITH_SETJMP
3018# define IEM_OPCODE_GET_NEXT_U64(a_pu64) \
3019 do \
3020 { \
3021 VBOXSTRICTRC rcStrict2 = iemOpcodeGetNextU64(pVCpu, (a_pu64)); \
3022 if (rcStrict2 != VINF_SUCCESS) \
3023 return rcStrict2; \
3024 } while (0)
3025#else
3026# define IEM_OPCODE_GET_NEXT_U64(a_pu64) ( *(a_pu64) = iemOpcodeGetNextU64Jmp(pVCpu) )
3027#endif
3028
3029
3030/** @name Misc Worker Functions.
3031 * @{
3032 */
3033
3034
3035/**
3036 * Validates a new SS segment.
3037 *
3038 * @returns VBox strict status code.
3039 * @param pVCpu The cross context virtual CPU structure of the
3040 * calling thread.
3041 * @param pCtx The CPU context.
3042 * @param NewSS The new SS selctor.
3043 * @param uCpl The CPL to load the stack for.
3044 * @param pDesc Where to return the descriptor.
3045 */
3046IEM_STATIC VBOXSTRICTRC iemMiscValidateNewSS(PVMCPU pVCpu, PCCPUMCTX pCtx, RTSEL NewSS, uint8_t uCpl, PIEMSELDESC pDesc)
3047{
3048 NOREF(pCtx);
3049
3050 /* Null selectors are not allowed (we're not called for dispatching
3051 interrupts with SS=0 in long mode). */
3052 if (!(NewSS & X86_SEL_MASK_OFF_RPL))
3053 {
3054 Log(("iemMiscValidateNewSSandRsp: %#x - null selector -> #TS(0)\n", NewSS));
3055 return iemRaiseTaskSwitchFault0(pVCpu);
3056 }
3057
3058 /** @todo testcase: check that the TSS.ssX RPL is checked. Also check when. */
3059 if ((NewSS & X86_SEL_RPL) != uCpl)
3060 {
3061 Log(("iemMiscValidateNewSSandRsp: %#x - RPL and CPL (%d) differs -> #TS\n", NewSS, uCpl));
3062 return iemRaiseTaskSwitchFaultBySelector(pVCpu, NewSS);
3063 }
3064
3065 /*
3066 * Read the descriptor.
3067 */
3068 VBOXSTRICTRC rcStrict = iemMemFetchSelDesc(pVCpu, pDesc, NewSS, X86_XCPT_TS);
3069 if (rcStrict != VINF_SUCCESS)
3070 return rcStrict;
3071
3072 /*
3073 * Perform the descriptor validation documented for LSS, POP SS and MOV SS.
3074 */
3075 if (!pDesc->Legacy.Gen.u1DescType)
3076 {
3077 Log(("iemMiscValidateNewSSandRsp: %#x - system selector (%#x) -> #TS\n", NewSS, pDesc->Legacy.Gen.u4Type));
3078 return iemRaiseTaskSwitchFaultBySelector(pVCpu, NewSS);
3079 }
3080
3081 if ( (pDesc->Legacy.Gen.u4Type & X86_SEL_TYPE_CODE)
3082 || !(pDesc->Legacy.Gen.u4Type & X86_SEL_TYPE_WRITE) )
3083 {
3084 Log(("iemMiscValidateNewSSandRsp: %#x - code or read only (%#x) -> #TS\n", NewSS, pDesc->Legacy.Gen.u4Type));
3085 return iemRaiseTaskSwitchFaultBySelector(pVCpu, NewSS);
3086 }
3087 if (pDesc->Legacy.Gen.u2Dpl != uCpl)
3088 {
3089 Log(("iemMiscValidateNewSSandRsp: %#x - DPL (%d) and CPL (%d) differs -> #TS\n", NewSS, pDesc->Legacy.Gen.u2Dpl, uCpl));
3090 return iemRaiseTaskSwitchFaultBySelector(pVCpu, NewSS);
3091 }
3092
3093 /* Is it there? */
3094 /** @todo testcase: Is this checked before the canonical / limit check below? */
3095 if (!pDesc->Legacy.Gen.u1Present)
3096 {
3097 Log(("iemMiscValidateNewSSandRsp: %#x - segment not present -> #NP\n", NewSS));
3098 return iemRaiseSelectorNotPresentBySelector(pVCpu, NewSS);
3099 }
3100
3101 return VINF_SUCCESS;
3102}
3103
3104
3105/**
3106 * Gets the correct EFLAGS regardless of whether PATM stores parts of them or
3107 * not.
3108 *
3109 * @param a_pVCpu The cross context virtual CPU structure of the calling thread.
3110 * @param a_pCtx The CPU context.
3111 */
3112#ifdef VBOX_WITH_RAW_MODE_NOT_R0
3113# define IEMMISC_GET_EFL(a_pVCpu, a_pCtx) \
3114 ( IEM_VERIFICATION_ENABLED(a_pVCpu) \
3115 ? (a_pCtx)->eflags.u \
3116 : CPUMRawGetEFlags(a_pVCpu) )
3117#else
3118# define IEMMISC_GET_EFL(a_pVCpu, a_pCtx) \
3119 ( (a_pCtx)->eflags.u )
3120#endif
3121
3122/**
3123 * Updates the EFLAGS in the correct manner wrt. PATM.
3124 *
3125 * @param a_pVCpu The cross context virtual CPU structure of the calling thread.
3126 * @param a_pCtx The CPU context.
3127 * @param a_fEfl The new EFLAGS.
3128 */
3129#ifdef VBOX_WITH_RAW_MODE_NOT_R0
3130# define IEMMISC_SET_EFL(a_pVCpu, a_pCtx, a_fEfl) \
3131 do { \
3132 if (IEM_VERIFICATION_ENABLED(a_pVCpu)) \
3133 (a_pCtx)->eflags.u = (a_fEfl); \
3134 else \
3135 CPUMRawSetEFlags((a_pVCpu), a_fEfl); \
3136 } while (0)
3137#else
3138# define IEMMISC_SET_EFL(a_pVCpu, a_pCtx, a_fEfl) \
3139 do { \
3140 (a_pCtx)->eflags.u = (a_fEfl); \
3141 } while (0)
3142#endif
3143
3144
3145/** @} */
3146
3147/** @name Raising Exceptions.
3148 *
3149 * @{
3150 */
3151
3152/** @name IEM_XCPT_FLAGS_XXX - flags for iemRaiseXcptOrInt.
3153 * @{ */
3154/** CPU exception. */
3155#define IEM_XCPT_FLAGS_T_CPU_XCPT RT_BIT_32(0)
3156/** External interrupt (from PIC, APIC, whatever). */
3157#define IEM_XCPT_FLAGS_T_EXT_INT RT_BIT_32(1)
3158/** Software interrupt (int or into, not bound).
3159 * Returns to the following instruction */
3160#define IEM_XCPT_FLAGS_T_SOFT_INT RT_BIT_32(2)
3161/** Takes an error code. */
3162#define IEM_XCPT_FLAGS_ERR RT_BIT_32(3)
3163/** Takes a CR2. */
3164#define IEM_XCPT_FLAGS_CR2 RT_BIT_32(4)
3165/** Generated by the breakpoint instruction. */
3166#define IEM_XCPT_FLAGS_BP_INSTR RT_BIT_32(5)
3167/** Generated by a DRx instruction breakpoint and RF should be cleared. */
3168#define IEM_XCPT_FLAGS_DRx_INSTR_BP RT_BIT_32(6)
3169/** @} */
3170
3171
3172/**
3173 * Loads the specified stack far pointer from the TSS.
3174 *
3175 * @returns VBox strict status code.
3176 * @param pVCpu The cross context virtual CPU structure of the calling thread.
3177 * @param pCtx The CPU context.
3178 * @param uCpl The CPL to load the stack for.
3179 * @param pSelSS Where to return the new stack segment.
3180 * @param puEsp Where to return the new stack pointer.
3181 */
3182IEM_STATIC VBOXSTRICTRC iemRaiseLoadStackFromTss32Or16(PVMCPU pVCpu, PCCPUMCTX pCtx, uint8_t uCpl,
3183 PRTSEL pSelSS, uint32_t *puEsp)
3184{
3185 VBOXSTRICTRC rcStrict;
3186 Assert(uCpl < 4);
3187
3188 switch (pCtx->tr.Attr.n.u4Type)
3189 {
3190 /*
3191 * 16-bit TSS (X86TSS16).
3192 */
3193 case X86_SEL_TYPE_SYS_286_TSS_AVAIL: AssertFailed();
3194 case X86_SEL_TYPE_SYS_286_TSS_BUSY:
3195 {
3196 uint32_t off = uCpl * 4 + 2;
3197 if (off + 4 <= pCtx->tr.u32Limit)
3198 {
3199 /** @todo check actual access pattern here. */
3200 uint32_t u32Tmp = 0; /* gcc maybe... */
3201 rcStrict = iemMemFetchSysU32(pVCpu, &u32Tmp, UINT8_MAX, pCtx->tr.u64Base + off);
3202 if (rcStrict == VINF_SUCCESS)
3203 {
3204 *puEsp = RT_LOWORD(u32Tmp);
3205 *pSelSS = RT_HIWORD(u32Tmp);
3206 return VINF_SUCCESS;
3207 }
3208 }
3209 else
3210 {
3211 Log(("LoadStackFromTss32Or16: out of bounds! uCpl=%d, u32Limit=%#x TSS16\n", uCpl, pCtx->tr.u32Limit));
3212 rcStrict = iemRaiseTaskSwitchFaultCurrentTSS(pVCpu);
3213 }
3214 break;
3215 }
3216
3217 /*
3218 * 32-bit TSS (X86TSS32).
3219 */
3220 case X86_SEL_TYPE_SYS_386_TSS_AVAIL: AssertFailed();
3221 case X86_SEL_TYPE_SYS_386_TSS_BUSY:
3222 {
3223 uint32_t off = uCpl * 8 + 4;
3224 if (off + 7 <= pCtx->tr.u32Limit)
3225 {
3226/** @todo check actual access pattern here. */
3227 uint64_t u64Tmp;
3228 rcStrict = iemMemFetchSysU64(pVCpu, &u64Tmp, UINT8_MAX, pCtx->tr.u64Base + off);
3229 if (rcStrict == VINF_SUCCESS)
3230 {
3231 *puEsp = u64Tmp & UINT32_MAX;
3232 *pSelSS = (RTSEL)(u64Tmp >> 32);
3233 return VINF_SUCCESS;
3234 }
3235 }
3236 else
3237 {
3238 Log(("LoadStackFromTss32Or16: out of bounds! uCpl=%d, u32Limit=%#x TSS16\n", uCpl, pCtx->tr.u32Limit));
3239 rcStrict = iemRaiseTaskSwitchFaultCurrentTSS(pVCpu);
3240 }
3241 break;
3242 }
3243
3244 default:
3245 AssertFailed();
3246 rcStrict = VERR_IEM_IPE_4;
3247 break;
3248 }
3249
3250 *puEsp = 0; /* make gcc happy */
3251 *pSelSS = 0; /* make gcc happy */
3252 return rcStrict;
3253}
3254
3255
3256/**
3257 * Loads the specified stack pointer from the 64-bit TSS.
3258 *
3259 * @returns VBox strict status code.
3260 * @param pVCpu The cross context virtual CPU structure of the calling thread.
3261 * @param pCtx The CPU context.
3262 * @param uCpl The CPL to load the stack for.
3263 * @param uIst The interrupt stack table index, 0 if to use uCpl.
3264 * @param puRsp Where to return the new stack pointer.
3265 */
3266IEM_STATIC VBOXSTRICTRC iemRaiseLoadStackFromTss64(PVMCPU pVCpu, PCCPUMCTX pCtx, uint8_t uCpl, uint8_t uIst, uint64_t *puRsp)
3267{
3268 Assert(uCpl < 4);
3269 Assert(uIst < 8);
3270 *puRsp = 0; /* make gcc happy */
3271
3272 AssertReturn(pCtx->tr.Attr.n.u4Type == AMD64_SEL_TYPE_SYS_TSS_BUSY, VERR_IEM_IPE_5);
3273
3274 uint32_t off;
3275 if (uIst)
3276 off = (uIst - 1) * sizeof(uint64_t) + RT_OFFSETOF(X86TSS64, ist1);
3277 else
3278 off = uCpl * sizeof(uint64_t) + RT_OFFSETOF(X86TSS64, rsp0);
3279 if (off + sizeof(uint64_t) > pCtx->tr.u32Limit)
3280 {
3281 Log(("iemRaiseLoadStackFromTss64: out of bounds! uCpl=%d uIst=%d, u32Limit=%#x\n", uCpl, uIst, pCtx->tr.u32Limit));
3282 return iemRaiseTaskSwitchFaultCurrentTSS(pVCpu);
3283 }
3284
3285 return iemMemFetchSysU64(pVCpu, puRsp, UINT8_MAX, pCtx->tr.u64Base + off);
3286}
3287
3288
3289/**
3290 * Adjust the CPU state according to the exception being raised.
3291 *
3292 * @param pCtx The CPU context.
3293 * @param u8Vector The exception that has been raised.
3294 */
3295DECLINLINE(void) iemRaiseXcptAdjustState(PCPUMCTX pCtx, uint8_t u8Vector)
3296{
3297 switch (u8Vector)
3298 {
3299 case X86_XCPT_DB:
3300 pCtx->dr[7] &= ~X86_DR7_GD;
3301 break;
3302 /** @todo Read the AMD and Intel exception reference... */
3303 }
3304}
3305
3306
3307/**
3308 * Implements exceptions and interrupts for real mode.
3309 *
3310 * @returns VBox strict status code.
3311 * @param pVCpu The cross context virtual CPU structure of the calling thread.
3312 * @param pCtx The CPU context.
3313 * @param cbInstr The number of bytes to offset rIP by in the return
3314 * address.
3315 * @param u8Vector The interrupt / exception vector number.
3316 * @param fFlags The flags.
3317 * @param uErr The error value if IEM_XCPT_FLAGS_ERR is set.
3318 * @param uCr2 The CR2 value if IEM_XCPT_FLAGS_CR2 is set.
3319 */
3320IEM_STATIC VBOXSTRICTRC
3321iemRaiseXcptOrIntInRealMode(PVMCPU pVCpu,
3322 PCPUMCTX pCtx,
3323 uint8_t cbInstr,
3324 uint8_t u8Vector,
3325 uint32_t fFlags,
3326 uint16_t uErr,
3327 uint64_t uCr2)
3328{
3329 AssertReturn(pVCpu->iem.s.enmCpuMode == IEMMODE_16BIT, VERR_IEM_IPE_6);
3330 NOREF(uErr); NOREF(uCr2);
3331
3332 /*
3333 * Read the IDT entry.
3334 */
3335 if (pCtx->idtr.cbIdt < UINT32_C(4) * u8Vector + 3)
3336 {
3337 Log(("RaiseXcptOrIntInRealMode: %#x is out of bounds (%#x)\n", u8Vector, pCtx->idtr.cbIdt));
3338 return iemRaiseGeneralProtectionFault(pVCpu, X86_TRAP_ERR_IDT | ((uint16_t)u8Vector << X86_TRAP_ERR_SEL_SHIFT));
3339 }
3340 RTFAR16 Idte;
3341 VBOXSTRICTRC rcStrict = iemMemFetchDataU32(pVCpu, (uint32_t *)&Idte, UINT8_MAX, pCtx->idtr.pIdt + UINT32_C(4) * u8Vector);
3342 if (RT_UNLIKELY(rcStrict != VINF_SUCCESS))
3343 return rcStrict;
3344
3345 /*
3346 * Push the stack frame.
3347 */
3348 uint16_t *pu16Frame;
3349 uint64_t uNewRsp;
3350 rcStrict = iemMemStackPushBeginSpecial(pVCpu, 6, (void **)&pu16Frame, &uNewRsp);
3351 if (rcStrict != VINF_SUCCESS)
3352 return rcStrict;
3353
3354 uint32_t fEfl = IEMMISC_GET_EFL(pVCpu, pCtx);
3355#if IEM_CFG_TARGET_CPU == IEMTARGETCPU_DYNAMIC
3356 AssertCompile(IEMTARGETCPU_8086 <= IEMTARGETCPU_186 && IEMTARGETCPU_V20 <= IEMTARGETCPU_186 && IEMTARGETCPU_286 > IEMTARGETCPU_186);
3357 if (pVCpu->iem.s.uTargetCpu <= IEMTARGETCPU_186)
3358 fEfl |= UINT16_C(0xf000);
3359#endif
3360 pu16Frame[2] = (uint16_t)fEfl;
3361 pu16Frame[1] = (uint16_t)pCtx->cs.Sel;
3362 pu16Frame[0] = (fFlags & IEM_XCPT_FLAGS_T_SOFT_INT) ? pCtx->ip + cbInstr : pCtx->ip;
3363 rcStrict = iemMemStackPushCommitSpecial(pVCpu, pu16Frame, uNewRsp);
3364 if (RT_UNLIKELY(rcStrict != VINF_SUCCESS))
3365 return rcStrict;
3366
3367 /*
3368 * Load the vector address into cs:ip and make exception specific state
3369 * adjustments.
3370 */
3371 pCtx->cs.Sel = Idte.sel;
3372 pCtx->cs.ValidSel = Idte.sel;
3373 pCtx->cs.fFlags = CPUMSELREG_FLAGS_VALID;
3374 pCtx->cs.u64Base = (uint32_t)Idte.sel << 4;
3375 /** @todo do we load attribs and limit as well? Should we check against limit like far jump? */
3376 pCtx->rip = Idte.off;
3377 fEfl &= ~X86_EFL_IF;
3378 IEMMISC_SET_EFL(pVCpu, pCtx, fEfl);
3379
3380 /** @todo do we actually do this in real mode? */
3381 if (fFlags & IEM_XCPT_FLAGS_T_CPU_XCPT)
3382 iemRaiseXcptAdjustState(pCtx, u8Vector);
3383
3384 return fFlags & IEM_XCPT_FLAGS_T_CPU_XCPT ? VINF_IEM_RAISED_XCPT : VINF_SUCCESS;
3385}
3386
3387
3388/**
3389 * Loads a NULL data selector into when coming from V8086 mode.
3390 *
3391 * @param pVCpu The cross context virtual CPU structure of the calling thread.
3392 * @param pSReg Pointer to the segment register.
3393 */
3394IEM_STATIC void iemHlpLoadNullDataSelectorOnV86Xcpt(PVMCPU pVCpu, PCPUMSELREG pSReg)
3395{
3396 pSReg->Sel = 0;
3397 pSReg->ValidSel = 0;
3398 if (IEM_IS_GUEST_CPU_INTEL(pVCpu) && !IEM_FULL_VERIFICATION_REM_ENABLED(pVCpu))
3399 {
3400 /* VT-x (Intel 3960x) doesn't change the base and limit, clears and sets the following attributes */
3401 pSReg->Attr.u &= X86DESCATTR_DT | X86DESCATTR_TYPE | X86DESCATTR_DPL | X86DESCATTR_G | X86DESCATTR_D;
3402 pSReg->Attr.u |= X86DESCATTR_UNUSABLE;
3403 }
3404 else
3405 {
3406 pSReg->fFlags = CPUMSELREG_FLAGS_VALID;
3407 /** @todo check this on AMD-V */
3408 pSReg->u64Base = 0;
3409 pSReg->u32Limit = 0;
3410 }
3411}
3412
3413
3414/**
3415 * Loads a segment selector during a task switch in V8086 mode.
3416 *
3417 * @param pSReg Pointer to the segment register.
3418 * @param uSel The selector value to load.
3419 */
3420IEM_STATIC void iemHlpLoadSelectorInV86Mode(PCPUMSELREG pSReg, uint16_t uSel)
3421{
3422 /* See Intel spec. 26.3.1.2 "Checks on Guest Segment Registers". */
3423 pSReg->Sel = uSel;
3424 pSReg->ValidSel = uSel;
3425 pSReg->fFlags = CPUMSELREG_FLAGS_VALID;
3426 pSReg->u64Base = uSel << 4;
3427 pSReg->u32Limit = 0xffff;
3428 pSReg->Attr.u = 0xf3;
3429}
3430
3431
3432/**
3433 * Loads a NULL data selector into a selector register, both the hidden and
3434 * visible parts, in protected mode.
3435 *
3436 * @param pVCpu The cross context virtual CPU structure of the calling thread.
3437 * @param pSReg Pointer to the segment register.
3438 * @param uRpl The RPL.
3439 */
3440IEM_STATIC void iemHlpLoadNullDataSelectorProt(PVMCPU pVCpu, PCPUMSELREG pSReg, RTSEL uRpl)
3441{
3442 /** @todo Testcase: write a testcase checking what happends when loading a NULL
3443 * data selector in protected mode. */
3444 pSReg->Sel = uRpl;
3445 pSReg->ValidSel = uRpl;
3446 pSReg->fFlags = CPUMSELREG_FLAGS_VALID;
3447 if (IEM_IS_GUEST_CPU_INTEL(pVCpu) && !IEM_FULL_VERIFICATION_REM_ENABLED(pVCpu))
3448 {
3449 /* VT-x (Intel 3960x) observed doing something like this. */
3450 pSReg->Attr.u = X86DESCATTR_UNUSABLE | X86DESCATTR_G | X86DESCATTR_D | (pVCpu->iem.s.uCpl << X86DESCATTR_DPL_SHIFT);
3451 pSReg->u32Limit = UINT32_MAX;
3452 pSReg->u64Base = 0;
3453 }
3454 else
3455 {
3456 pSReg->Attr.u = X86DESCATTR_UNUSABLE;
3457 pSReg->u32Limit = 0;
3458 pSReg->u64Base = 0;
3459 }
3460}
3461
3462
3463/**
3464 * Loads a segment selector during a task switch in protected mode.
3465 *
3466 * In this task switch scenario, we would throw \#TS exceptions rather than
3467 * \#GPs.
3468 *
3469 * @returns VBox strict status code.
3470 * @param pVCpu The cross context virtual CPU structure of the calling thread.
3471 * @param pSReg Pointer to the segment register.
3472 * @param uSel The new selector value.
3473 *
3474 * @remarks This does _not_ handle CS or SS.
3475 * @remarks This expects pVCpu->iem.s.uCpl to be up to date.
3476 */
3477IEM_STATIC VBOXSTRICTRC iemHlpTaskSwitchLoadDataSelectorInProtMode(PVMCPU pVCpu, PCPUMSELREG pSReg, uint16_t uSel)
3478{
3479 Assert(pVCpu->iem.s.enmCpuMode != IEMMODE_64BIT);
3480
3481 /* Null data selector. */
3482 if (!(uSel & X86_SEL_MASK_OFF_RPL))
3483 {
3484 iemHlpLoadNullDataSelectorProt(pVCpu, pSReg, uSel);
3485 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, pSReg));
3486 CPUMSetChangedFlags(pVCpu, CPUM_CHANGED_HIDDEN_SEL_REGS);
3487 return VINF_SUCCESS;
3488 }
3489
3490 /* Fetch the descriptor. */
3491 IEMSELDESC Desc;
3492 VBOXSTRICTRC rcStrict = iemMemFetchSelDesc(pVCpu, &Desc, uSel, X86_XCPT_TS);
3493 if (rcStrict != VINF_SUCCESS)
3494 {
3495 Log(("iemHlpTaskSwitchLoadDataSelectorInProtMode: failed to fetch selector. uSel=%u rc=%Rrc\n", uSel,
3496 VBOXSTRICTRC_VAL(rcStrict)));
3497 return rcStrict;
3498 }
3499
3500 /* Must be a data segment or readable code segment. */
3501 if ( !Desc.Legacy.Gen.u1DescType
3502 || (Desc.Legacy.Gen.u4Type & (X86_SEL_TYPE_CODE | X86_SEL_TYPE_READ)) == X86_SEL_TYPE_CODE)
3503 {
3504 Log(("iemHlpTaskSwitchLoadDataSelectorInProtMode: invalid segment type. uSel=%u Desc.u4Type=%#x\n", uSel,
3505 Desc.Legacy.Gen.u4Type));
3506 return iemRaiseTaskSwitchFaultWithErr(pVCpu, uSel & X86_SEL_MASK_OFF_RPL);
3507 }
3508
3509 /* Check privileges for data segments and non-conforming code segments. */
3510 if ( (Desc.Legacy.Gen.u4Type & (X86_SEL_TYPE_CODE | X86_SEL_TYPE_CONF))
3511 != (X86_SEL_TYPE_CODE | X86_SEL_TYPE_CONF))
3512 {
3513 /* The RPL and the new CPL must be less than or equal to the DPL. */
3514 if ( (unsigned)(uSel & X86_SEL_RPL) > Desc.Legacy.Gen.u2Dpl
3515 || (pVCpu->iem.s.uCpl > Desc.Legacy.Gen.u2Dpl))
3516 {
3517 Log(("iemHlpTaskSwitchLoadDataSelectorInProtMode: Invalid priv. uSel=%u uSel.RPL=%u DPL=%u CPL=%u\n",
3518 uSel, (uSel & X86_SEL_RPL), Desc.Legacy.Gen.u2Dpl, pVCpu->iem.s.uCpl));
3519 return iemRaiseTaskSwitchFaultWithErr(pVCpu, uSel & X86_SEL_MASK_OFF_RPL);
3520 }
3521 }
3522
3523 /* Is it there? */
3524 if (!Desc.Legacy.Gen.u1Present)
3525 {
3526 Log(("iemHlpTaskSwitchLoadDataSelectorInProtMode: Segment not present. uSel=%u\n", uSel));
3527 return iemRaiseSelectorNotPresentWithErr(pVCpu, uSel & X86_SEL_MASK_OFF_RPL);
3528 }
3529
3530 /* The base and limit. */
3531 uint32_t cbLimit = X86DESC_LIMIT_G(&Desc.Legacy);
3532 uint64_t u64Base = X86DESC_BASE(&Desc.Legacy);
3533
3534 /*
3535 * Ok, everything checked out fine. Now set the accessed bit before
3536 * committing the result into the registers.
3537 */
3538 if (!(Desc.Legacy.Gen.u4Type & X86_SEL_TYPE_ACCESSED))
3539 {
3540 rcStrict = iemMemMarkSelDescAccessed(pVCpu, uSel);
3541 if (rcStrict != VINF_SUCCESS)
3542 return rcStrict;
3543 Desc.Legacy.Gen.u4Type |= X86_SEL_TYPE_ACCESSED;
3544 }
3545
3546 /* Commit */
3547 pSReg->Sel = uSel;
3548 pSReg->Attr.u = X86DESC_GET_HID_ATTR(&Desc.Legacy);
3549 pSReg->u32Limit = cbLimit;
3550 pSReg->u64Base = u64Base; /** @todo testcase/investigate: seen claims that the upper half of the base remains unchanged... */
3551 pSReg->ValidSel = uSel;
3552 pSReg->fFlags = CPUMSELREG_FLAGS_VALID;
3553 if (IEM_IS_GUEST_CPU_INTEL(pVCpu) && !IEM_FULL_VERIFICATION_REM_ENABLED(pVCpu))
3554 pSReg->Attr.u &= ~X86DESCATTR_UNUSABLE;
3555
3556 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, pSReg));
3557 CPUMSetChangedFlags(pVCpu, CPUM_CHANGED_HIDDEN_SEL_REGS);
3558 return VINF_SUCCESS;
3559}
3560
3561
3562/**
3563 * Performs a task switch.
3564 *
3565 * If the task switch is the result of a JMP, CALL or IRET instruction, the
3566 * caller is responsible for performing the necessary checks (like DPL, TSS
3567 * present etc.) which are specific to JMP/CALL/IRET. See Intel Instruction
3568 * reference for JMP, CALL, IRET.
3569 *
3570 * If the task switch is the due to a software interrupt or hardware exception,
3571 * the caller is responsible for validating the TSS selector and descriptor. See
3572 * Intel Instruction reference for INT n.
3573 *
3574 * @returns VBox strict status code.
3575 * @param pVCpu The cross context virtual CPU structure of the calling thread.
3576 * @param pCtx The CPU context.
3577 * @param enmTaskSwitch What caused this task switch.
3578 * @param uNextEip The EIP effective after the task switch.
3579 * @param fFlags The flags.
3580 * @param uErr The error value if IEM_XCPT_FLAGS_ERR is set.
3581 * @param uCr2 The CR2 value if IEM_XCPT_FLAGS_CR2 is set.
3582 * @param SelTSS The TSS selector of the new task.
3583 * @param pNewDescTSS Pointer to the new TSS descriptor.
3584 */
3585IEM_STATIC VBOXSTRICTRC
3586iemTaskSwitch(PVMCPU pVCpu,
3587 PCPUMCTX pCtx,
3588 IEMTASKSWITCH enmTaskSwitch,
3589 uint32_t uNextEip,
3590 uint32_t fFlags,
3591 uint16_t uErr,
3592 uint64_t uCr2,
3593 RTSEL SelTSS,
3594 PIEMSELDESC pNewDescTSS)
3595{
3596 Assert(!IEM_IS_REAL_MODE(pVCpu));
3597 Assert(pVCpu->iem.s.enmCpuMode != IEMMODE_64BIT);
3598
3599 uint32_t const uNewTSSType = pNewDescTSS->Legacy.Gate.u4Type;
3600 Assert( uNewTSSType == X86_SEL_TYPE_SYS_286_TSS_AVAIL
3601 || uNewTSSType == X86_SEL_TYPE_SYS_286_TSS_BUSY
3602 || uNewTSSType == X86_SEL_TYPE_SYS_386_TSS_AVAIL
3603 || uNewTSSType == X86_SEL_TYPE_SYS_386_TSS_BUSY);
3604
3605 bool const fIsNewTSS386 = ( uNewTSSType == X86_SEL_TYPE_SYS_386_TSS_AVAIL
3606 || uNewTSSType == X86_SEL_TYPE_SYS_386_TSS_BUSY);
3607
3608 Log(("iemTaskSwitch: enmTaskSwitch=%u NewTSS=%#x fIsNewTSS386=%RTbool EIP=%#RX32 uNextEip=%#RX32\n", enmTaskSwitch, SelTSS,
3609 fIsNewTSS386, pCtx->eip, uNextEip));
3610
3611 /* Update CR2 in case it's a page-fault. */
3612 /** @todo This should probably be done much earlier in IEM/PGM. See
3613 * @bugref{5653#c49}. */
3614 if (fFlags & IEM_XCPT_FLAGS_CR2)
3615 pCtx->cr2 = uCr2;
3616
3617 /*
3618 * Check the new TSS limit. See Intel spec. 6.15 "Exception and Interrupt Reference"
3619 * subsection "Interrupt 10 - Invalid TSS Exception (#TS)".
3620 */
3621 uint32_t const uNewTSSLimit = pNewDescTSS->Legacy.Gen.u16LimitLow | (pNewDescTSS->Legacy.Gen.u4LimitHigh << 16);
3622 uint32_t const uNewTSSLimitMin = fIsNewTSS386 ? X86_SEL_TYPE_SYS_386_TSS_LIMIT_MIN : X86_SEL_TYPE_SYS_286_TSS_LIMIT_MIN;
3623 if (uNewTSSLimit < uNewTSSLimitMin)
3624 {
3625 Log(("iemTaskSwitch: Invalid new TSS limit. enmTaskSwitch=%u uNewTSSLimit=%#x uNewTSSLimitMin=%#x -> #TS\n",
3626 enmTaskSwitch, uNewTSSLimit, uNewTSSLimitMin));
3627 return iemRaiseTaskSwitchFaultWithErr(pVCpu, SelTSS & X86_SEL_MASK_OFF_RPL);
3628 }
3629
3630 /*
3631 * Check the current TSS limit. The last written byte to the current TSS during the
3632 * task switch will be 2 bytes at offset 0x5C (32-bit) and 1 byte at offset 0x28 (16-bit).
3633 * See Intel spec. 7.2.1 "Task-State Segment (TSS)" for static and dynamic fields.
3634 *
3635 * The AMD docs doesn't mention anything about limit checks with LTR which suggests you can
3636 * end up with smaller than "legal" TSS limits.
3637 */
3638 uint32_t const uCurTSSLimit = pCtx->tr.u32Limit;
3639 uint32_t const uCurTSSLimitMin = fIsNewTSS386 ? 0x5F : 0x29;
3640 if (uCurTSSLimit < uCurTSSLimitMin)
3641 {
3642 Log(("iemTaskSwitch: Invalid current TSS limit. enmTaskSwitch=%u uCurTSSLimit=%#x uCurTSSLimitMin=%#x -> #TS\n",
3643 enmTaskSwitch, uCurTSSLimit, uCurTSSLimitMin));
3644 return iemRaiseTaskSwitchFaultWithErr(pVCpu, SelTSS & X86_SEL_MASK_OFF_RPL);
3645 }
3646
3647 /*
3648 * Verify that the new TSS can be accessed and map it. Map only the required contents
3649 * and not the entire TSS.
3650 */
3651 void *pvNewTSS;
3652 uint32_t cbNewTSS = uNewTSSLimitMin + 1;
3653 RTGCPTR GCPtrNewTSS = X86DESC_BASE(&pNewDescTSS->Legacy);
3654 AssertCompile(sizeof(X86TSS32) == X86_SEL_TYPE_SYS_386_TSS_LIMIT_MIN + 1);
3655 /** @todo Handle if the TSS crosses a page boundary. Intel specifies that it may
3656 * not perform correct translation if this happens. See Intel spec. 7.2.1
3657 * "Task-State Segment" */
3658 VBOXSTRICTRC rcStrict = iemMemMap(pVCpu, &pvNewTSS, cbNewTSS, UINT8_MAX, GCPtrNewTSS, IEM_ACCESS_SYS_RW);
3659 if (rcStrict != VINF_SUCCESS)
3660 {
3661 Log(("iemTaskSwitch: Failed to read new TSS. enmTaskSwitch=%u cbNewTSS=%u uNewTSSLimit=%u rc=%Rrc\n", enmTaskSwitch,
3662 cbNewTSS, uNewTSSLimit, VBOXSTRICTRC_VAL(rcStrict)));
3663 return rcStrict;
3664 }
3665
3666 /*
3667 * Clear the busy bit in current task's TSS descriptor if it's a task switch due to JMP/IRET.
3668 */
3669 uint32_t u32EFlags = pCtx->eflags.u32;
3670 if ( enmTaskSwitch == IEMTASKSWITCH_JUMP
3671 || enmTaskSwitch == IEMTASKSWITCH_IRET)
3672 {
3673 PX86DESC pDescCurTSS;
3674 rcStrict = iemMemMap(pVCpu, (void **)&pDescCurTSS, sizeof(*pDescCurTSS), UINT8_MAX,
3675 pCtx->gdtr.pGdt + (pCtx->tr.Sel & X86_SEL_MASK), IEM_ACCESS_SYS_RW);
3676 if (rcStrict != VINF_SUCCESS)
3677 {
3678 Log(("iemTaskSwitch: Failed to read new TSS descriptor in GDT. enmTaskSwitch=%u pGdt=%#RX64 rc=%Rrc\n",
3679 enmTaskSwitch, pCtx->gdtr.pGdt, VBOXSTRICTRC_VAL(rcStrict)));
3680 return rcStrict;
3681 }
3682
3683 pDescCurTSS->Gate.u4Type &= ~X86_SEL_TYPE_SYS_TSS_BUSY_MASK;
3684 rcStrict = iemMemCommitAndUnmap(pVCpu, pDescCurTSS, IEM_ACCESS_SYS_RW);
3685 if (rcStrict != VINF_SUCCESS)
3686 {
3687 Log(("iemTaskSwitch: Failed to commit new TSS descriptor in GDT. enmTaskSwitch=%u pGdt=%#RX64 rc=%Rrc\n",
3688 enmTaskSwitch, pCtx->gdtr.pGdt, VBOXSTRICTRC_VAL(rcStrict)));
3689 return rcStrict;
3690 }
3691
3692 /* Clear EFLAGS.NT (Nested Task) in the eflags memory image, if it's a task switch due to an IRET. */
3693 if (enmTaskSwitch == IEMTASKSWITCH_IRET)
3694 {
3695 Assert( uNewTSSType == X86_SEL_TYPE_SYS_286_TSS_BUSY
3696 || uNewTSSType == X86_SEL_TYPE_SYS_386_TSS_BUSY);
3697 u32EFlags &= ~X86_EFL_NT;
3698 }
3699 }
3700
3701 /*
3702 * Save the CPU state into the current TSS.
3703 */
3704 RTGCPTR GCPtrCurTSS = pCtx->tr.u64Base;
3705 if (GCPtrNewTSS == GCPtrCurTSS)
3706 {
3707 Log(("iemTaskSwitch: Switching to the same TSS! enmTaskSwitch=%u GCPtr[Cur|New]TSS=%#RGv\n", enmTaskSwitch, GCPtrCurTSS));
3708 Log(("uCurCr3=%#x uCurEip=%#x uCurEflags=%#x uCurEax=%#x uCurEsp=%#x uCurEbp=%#x uCurCS=%#04x uCurSS=%#04x uCurLdt=%#x\n",
3709 pCtx->cr3, pCtx->eip, pCtx->eflags.u32, pCtx->eax, pCtx->esp, pCtx->ebp, pCtx->cs.Sel, pCtx->ss.Sel, pCtx->ldtr.Sel));
3710 }
3711 if (fIsNewTSS386)
3712 {
3713 /*
3714 * Verify that the current TSS (32-bit) can be accessed, only the minimum required size.
3715 * See Intel spec. 7.2.1 "Task-State Segment (TSS)" for static and dynamic fields.
3716 */
3717 void *pvCurTSS32;
3718 uint32_t offCurTSS = RT_OFFSETOF(X86TSS32, eip);
3719 uint32_t cbCurTSS = RT_OFFSETOF(X86TSS32, selLdt) - RT_OFFSETOF(X86TSS32, eip);
3720 AssertCompile(RTASSERT_OFFSET_OF(X86TSS32, selLdt) - RTASSERT_OFFSET_OF(X86TSS32, eip) == 64);
3721 rcStrict = iemMemMap(pVCpu, &pvCurTSS32, cbCurTSS, UINT8_MAX, GCPtrCurTSS + offCurTSS, IEM_ACCESS_SYS_RW);
3722 if (rcStrict != VINF_SUCCESS)
3723 {
3724 Log(("iemTaskSwitch: Failed to read current 32-bit TSS. enmTaskSwitch=%u GCPtrCurTSS=%#RGv cb=%u rc=%Rrc\n",
3725 enmTaskSwitch, GCPtrCurTSS, cbCurTSS, VBOXSTRICTRC_VAL(rcStrict)));
3726 return rcStrict;
3727 }
3728
3729 /* !! WARNING !! Access -only- the members (dynamic fields) that are mapped, i.e interval [offCurTSS..cbCurTSS). */
3730 PX86TSS32 pCurTSS32 = (PX86TSS32)((uintptr_t)pvCurTSS32 - offCurTSS);
3731 pCurTSS32->eip = uNextEip;
3732 pCurTSS32->eflags = u32EFlags;
3733 pCurTSS32->eax = pCtx->eax;
3734 pCurTSS32->ecx = pCtx->ecx;
3735 pCurTSS32->edx = pCtx->edx;
3736 pCurTSS32->ebx = pCtx->ebx;
3737 pCurTSS32->esp = pCtx->esp;
3738 pCurTSS32->ebp = pCtx->ebp;
3739 pCurTSS32->esi = pCtx->esi;
3740 pCurTSS32->edi = pCtx->edi;
3741 pCurTSS32->es = pCtx->es.Sel;
3742 pCurTSS32->cs = pCtx->cs.Sel;
3743 pCurTSS32->ss = pCtx->ss.Sel;
3744 pCurTSS32->ds = pCtx->ds.Sel;
3745 pCurTSS32->fs = pCtx->fs.Sel;
3746 pCurTSS32->gs = pCtx->gs.Sel;
3747
3748 rcStrict = iemMemCommitAndUnmap(pVCpu, pvCurTSS32, IEM_ACCESS_SYS_RW);
3749 if (rcStrict != VINF_SUCCESS)
3750 {
3751 Log(("iemTaskSwitch: Failed to commit current 32-bit TSS. enmTaskSwitch=%u rc=%Rrc\n", enmTaskSwitch,
3752 VBOXSTRICTRC_VAL(rcStrict)));
3753 return rcStrict;
3754 }
3755 }
3756 else
3757 {
3758 /*
3759 * Verify that the current TSS (16-bit) can be accessed. Again, only the minimum required size.
3760 */
3761 void *pvCurTSS16;
3762 uint32_t offCurTSS = RT_OFFSETOF(X86TSS16, ip);
3763 uint32_t cbCurTSS = RT_OFFSETOF(X86TSS16, selLdt) - RT_OFFSETOF(X86TSS16, ip);
3764 AssertCompile(RTASSERT_OFFSET_OF(X86TSS16, selLdt) - RTASSERT_OFFSET_OF(X86TSS16, ip) == 28);
3765 rcStrict = iemMemMap(pVCpu, &pvCurTSS16, cbCurTSS, UINT8_MAX, GCPtrCurTSS + offCurTSS, IEM_ACCESS_SYS_RW);
3766 if (rcStrict != VINF_SUCCESS)
3767 {
3768 Log(("iemTaskSwitch: Failed to read current 16-bit TSS. enmTaskSwitch=%u GCPtrCurTSS=%#RGv cb=%u rc=%Rrc\n",
3769 enmTaskSwitch, GCPtrCurTSS, cbCurTSS, VBOXSTRICTRC_VAL(rcStrict)));
3770 return rcStrict;
3771 }
3772
3773 /* !! WARNING !! Access -only- the members (dynamic fields) that are mapped, i.e interval [offCurTSS..cbCurTSS). */
3774 PX86TSS16 pCurTSS16 = (PX86TSS16)((uintptr_t)pvCurTSS16 - offCurTSS);
3775 pCurTSS16->ip = uNextEip;
3776 pCurTSS16->flags = u32EFlags;
3777 pCurTSS16->ax = pCtx->ax;
3778 pCurTSS16->cx = pCtx->cx;
3779 pCurTSS16->dx = pCtx->dx;
3780 pCurTSS16->bx = pCtx->bx;
3781 pCurTSS16->sp = pCtx->sp;
3782 pCurTSS16->bp = pCtx->bp;
3783 pCurTSS16->si = pCtx->si;
3784 pCurTSS16->di = pCtx->di;
3785 pCurTSS16->es = pCtx->es.Sel;
3786 pCurTSS16->cs = pCtx->cs.Sel;
3787 pCurTSS16->ss = pCtx->ss.Sel;
3788 pCurTSS16->ds = pCtx->ds.Sel;
3789
3790 rcStrict = iemMemCommitAndUnmap(pVCpu, pvCurTSS16, IEM_ACCESS_SYS_RW);
3791 if (rcStrict != VINF_SUCCESS)
3792 {
3793 Log(("iemTaskSwitch: Failed to commit current 16-bit TSS. enmTaskSwitch=%u rc=%Rrc\n", enmTaskSwitch,
3794 VBOXSTRICTRC_VAL(rcStrict)));
3795 return rcStrict;
3796 }
3797 }
3798
3799 /*
3800 * Update the previous task link field for the new TSS, if the task switch is due to a CALL/INT_XCPT.
3801 */
3802 if ( enmTaskSwitch == IEMTASKSWITCH_CALL
3803 || enmTaskSwitch == IEMTASKSWITCH_INT_XCPT)
3804 {
3805 /* 16 or 32-bit TSS doesn't matter, we only access the first, common 16-bit field (selPrev) here. */
3806 PX86TSS32 pNewTSS = (PX86TSS32)pvNewTSS;
3807 pNewTSS->selPrev = pCtx->tr.Sel;
3808 }
3809
3810 /*
3811 * Read the state from the new TSS into temporaries. Setting it immediately as the new CPU state is tricky,
3812 * it's done further below with error handling (e.g. CR3 changes will go through PGM).
3813 */
3814 uint32_t uNewCr3, uNewEip, uNewEflags, uNewEax, uNewEcx, uNewEdx, uNewEbx, uNewEsp, uNewEbp, uNewEsi, uNewEdi;
3815 uint16_t uNewES, uNewCS, uNewSS, uNewDS, uNewFS, uNewGS, uNewLdt;
3816 bool fNewDebugTrap;
3817 if (fIsNewTSS386)
3818 {
3819 PX86TSS32 pNewTSS32 = (PX86TSS32)pvNewTSS;
3820 uNewCr3 = (pCtx->cr0 & X86_CR0_PG) ? pNewTSS32->cr3 : 0;
3821 uNewEip = pNewTSS32->eip;
3822 uNewEflags = pNewTSS32->eflags;
3823 uNewEax = pNewTSS32->eax;
3824 uNewEcx = pNewTSS32->ecx;
3825 uNewEdx = pNewTSS32->edx;
3826 uNewEbx = pNewTSS32->ebx;
3827 uNewEsp = pNewTSS32->esp;
3828 uNewEbp = pNewTSS32->ebp;
3829 uNewEsi = pNewTSS32->esi;
3830 uNewEdi = pNewTSS32->edi;
3831 uNewES = pNewTSS32->es;
3832 uNewCS = pNewTSS32->cs;
3833 uNewSS = pNewTSS32->ss;
3834 uNewDS = pNewTSS32->ds;
3835 uNewFS = pNewTSS32->fs;
3836 uNewGS = pNewTSS32->gs;
3837 uNewLdt = pNewTSS32->selLdt;
3838 fNewDebugTrap = RT_BOOL(pNewTSS32->fDebugTrap);
3839 }
3840 else
3841 {
3842 PX86TSS16 pNewTSS16 = (PX86TSS16)pvNewTSS;
3843 uNewCr3 = 0;
3844 uNewEip = pNewTSS16->ip;
3845 uNewEflags = pNewTSS16->flags;
3846 uNewEax = UINT32_C(0xffff0000) | pNewTSS16->ax;
3847 uNewEcx = UINT32_C(0xffff0000) | pNewTSS16->cx;
3848 uNewEdx = UINT32_C(0xffff0000) | pNewTSS16->dx;
3849 uNewEbx = UINT32_C(0xffff0000) | pNewTSS16->bx;
3850 uNewEsp = UINT32_C(0xffff0000) | pNewTSS16->sp;
3851 uNewEbp = UINT32_C(0xffff0000) | pNewTSS16->bp;
3852 uNewEsi = UINT32_C(0xffff0000) | pNewTSS16->si;
3853 uNewEdi = UINT32_C(0xffff0000) | pNewTSS16->di;
3854 uNewES = pNewTSS16->es;
3855 uNewCS = pNewTSS16->cs;
3856 uNewSS = pNewTSS16->ss;
3857 uNewDS = pNewTSS16->ds;
3858 uNewFS = 0;
3859 uNewGS = 0;
3860 uNewLdt = pNewTSS16->selLdt;
3861 fNewDebugTrap = false;
3862 }
3863
3864 if (GCPtrNewTSS == GCPtrCurTSS)
3865 Log(("uNewCr3=%#x uNewEip=%#x uNewEflags=%#x uNewEax=%#x uNewEsp=%#x uNewEbp=%#x uNewCS=%#04x uNewSS=%#04x uNewLdt=%#x\n",
3866 uNewCr3, uNewEip, uNewEflags, uNewEax, uNewEsp, uNewEbp, uNewCS, uNewSS, uNewLdt));
3867
3868 /*
3869 * We're done accessing the new TSS.
3870 */
3871 rcStrict = iemMemCommitAndUnmap(pVCpu, pvNewTSS, IEM_ACCESS_SYS_RW);
3872 if (rcStrict != VINF_SUCCESS)
3873 {
3874 Log(("iemTaskSwitch: Failed to commit new TSS. enmTaskSwitch=%u rc=%Rrc\n", enmTaskSwitch, VBOXSTRICTRC_VAL(rcStrict)));
3875 return rcStrict;
3876 }
3877
3878 /*
3879 * Set the busy bit in the new TSS descriptor, if the task switch is a JMP/CALL/INT_XCPT.
3880 */
3881 if (enmTaskSwitch != IEMTASKSWITCH_IRET)
3882 {
3883 rcStrict = iemMemMap(pVCpu, (void **)&pNewDescTSS, sizeof(*pNewDescTSS), UINT8_MAX,
3884 pCtx->gdtr.pGdt + (SelTSS & X86_SEL_MASK), IEM_ACCESS_SYS_RW);
3885 if (rcStrict != VINF_SUCCESS)
3886 {
3887 Log(("iemTaskSwitch: Failed to read new TSS descriptor in GDT (2). enmTaskSwitch=%u pGdt=%#RX64 rc=%Rrc\n",
3888 enmTaskSwitch, pCtx->gdtr.pGdt, VBOXSTRICTRC_VAL(rcStrict)));
3889 return rcStrict;
3890 }
3891
3892 /* Check that the descriptor indicates the new TSS is available (not busy). */
3893 AssertMsg( pNewDescTSS->Legacy.Gate.u4Type == X86_SEL_TYPE_SYS_286_TSS_AVAIL
3894 || pNewDescTSS->Legacy.Gate.u4Type == X86_SEL_TYPE_SYS_386_TSS_AVAIL,
3895 ("Invalid TSS descriptor type=%#x", pNewDescTSS->Legacy.Gate.u4Type));
3896
3897 pNewDescTSS->Legacy.Gate.u4Type |= X86_SEL_TYPE_SYS_TSS_BUSY_MASK;
3898 rcStrict = iemMemCommitAndUnmap(pVCpu, pNewDescTSS, IEM_ACCESS_SYS_RW);
3899 if (rcStrict != VINF_SUCCESS)
3900 {
3901 Log(("iemTaskSwitch: Failed to commit new TSS descriptor in GDT (2). enmTaskSwitch=%u pGdt=%#RX64 rc=%Rrc\n",
3902 enmTaskSwitch, pCtx->gdtr.pGdt, VBOXSTRICTRC_VAL(rcStrict)));
3903 return rcStrict;
3904 }
3905 }
3906
3907 /*
3908 * From this point on, we're technically in the new task. We will defer exceptions
3909 * until the completion of the task switch but before executing any instructions in the new task.
3910 */
3911 pCtx->tr.Sel = SelTSS;
3912 pCtx->tr.ValidSel = SelTSS;
3913 pCtx->tr.fFlags = CPUMSELREG_FLAGS_VALID;
3914 pCtx->tr.Attr.u = X86DESC_GET_HID_ATTR(&pNewDescTSS->Legacy);
3915 pCtx->tr.u32Limit = X86DESC_LIMIT_G(&pNewDescTSS->Legacy);
3916 pCtx->tr.u64Base = X86DESC_BASE(&pNewDescTSS->Legacy);
3917 CPUMSetChangedFlags(pVCpu, CPUM_CHANGED_TR);
3918
3919 /* Set the busy bit in TR. */
3920 pCtx->tr.Attr.n.u4Type |= X86_SEL_TYPE_SYS_TSS_BUSY_MASK;
3921 /* Set EFLAGS.NT (Nested Task) in the eflags loaded from the new TSS, if it's a task switch due to a CALL/INT_XCPT. */
3922 if ( enmTaskSwitch == IEMTASKSWITCH_CALL
3923 || enmTaskSwitch == IEMTASKSWITCH_INT_XCPT)
3924 {
3925 uNewEflags |= X86_EFL_NT;
3926 }
3927
3928 pCtx->dr[7] &= ~X86_DR7_LE_ALL; /** @todo Should we clear DR7.LE bit too? */
3929 pCtx->cr0 |= X86_CR0_TS;
3930 CPUMSetChangedFlags(pVCpu, CPUM_CHANGED_CR0);
3931
3932 pCtx->eip = uNewEip;
3933 pCtx->eax = uNewEax;
3934 pCtx->ecx = uNewEcx;
3935 pCtx->edx = uNewEdx;
3936 pCtx->ebx = uNewEbx;
3937 pCtx->esp = uNewEsp;
3938 pCtx->ebp = uNewEbp;
3939 pCtx->esi = uNewEsi;
3940 pCtx->edi = uNewEdi;
3941
3942 uNewEflags &= X86_EFL_LIVE_MASK;
3943 uNewEflags |= X86_EFL_RA1_MASK;
3944 IEMMISC_SET_EFL(pVCpu, pCtx, uNewEflags);
3945
3946 /*
3947 * Switch the selectors here and do the segment checks later. If we throw exceptions, the selectors
3948 * will be valid in the exception handler. We cannot update the hidden parts until we've switched CR3
3949 * due to the hidden part data originating from the guest LDT/GDT which is accessed through paging.
3950 */
3951 pCtx->es.Sel = uNewES;
3952 pCtx->es.Attr.u &= ~X86DESCATTR_P;
3953
3954 pCtx->cs.Sel = uNewCS;
3955 pCtx->cs.Attr.u &= ~X86DESCATTR_P;
3956
3957 pCtx->ss.Sel = uNewSS;
3958 pCtx->ss.Attr.u &= ~X86DESCATTR_P;
3959
3960 pCtx->ds.Sel = uNewDS;
3961 pCtx->ds.Attr.u &= ~X86DESCATTR_P;
3962
3963 pCtx->fs.Sel = uNewFS;
3964 pCtx->fs.Attr.u &= ~X86DESCATTR_P;
3965
3966 pCtx->gs.Sel = uNewGS;
3967 pCtx->gs.Attr.u &= ~X86DESCATTR_P;
3968 CPUMSetChangedFlags(pVCpu, CPUM_CHANGED_HIDDEN_SEL_REGS);
3969
3970 pCtx->ldtr.Sel = uNewLdt;
3971 pCtx->ldtr.fFlags = CPUMSELREG_FLAGS_STALE;
3972 pCtx->ldtr.Attr.u &= ~X86DESCATTR_P;
3973 CPUMSetChangedFlags(pVCpu, CPUM_CHANGED_LDTR);
3974
3975 if (IEM_IS_GUEST_CPU_INTEL(pVCpu) && !IEM_FULL_VERIFICATION_REM_ENABLED(pVCpu))
3976 {
3977 pCtx->es.Attr.u |= X86DESCATTR_UNUSABLE;
3978 pCtx->cs.Attr.u |= X86DESCATTR_UNUSABLE;
3979 pCtx->ss.Attr.u |= X86DESCATTR_UNUSABLE;
3980 pCtx->ds.Attr.u |= X86DESCATTR_UNUSABLE;
3981 pCtx->fs.Attr.u |= X86DESCATTR_UNUSABLE;
3982 pCtx->gs.Attr.u |= X86DESCATTR_UNUSABLE;
3983 pCtx->ldtr.Attr.u |= X86DESCATTR_UNUSABLE;
3984 }
3985
3986 /*
3987 * Switch CR3 for the new task.
3988 */
3989 if ( fIsNewTSS386
3990 && (pCtx->cr0 & X86_CR0_PG))
3991 {
3992 /** @todo Should we update and flush TLBs only if CR3 value actually changes? */
3993 if (!IEM_FULL_VERIFICATION_ENABLED(pVCpu))
3994 {
3995 int rc = CPUMSetGuestCR3(pVCpu, uNewCr3);
3996 AssertRCSuccessReturn(rc, rc);
3997 }
3998 else
3999 pCtx->cr3 = uNewCr3;
4000
4001 /* Inform PGM. */
4002 if (!IEM_FULL_VERIFICATION_ENABLED(pVCpu))
4003 {
4004 int rc = PGMFlushTLB(pVCpu, pCtx->cr3, !(pCtx->cr4 & X86_CR4_PGE));
4005 AssertRCReturn(rc, rc);
4006 /* ignore informational status codes */
4007 }
4008 CPUMSetChangedFlags(pVCpu, CPUM_CHANGED_CR3);
4009 }
4010
4011 /*
4012 * Switch LDTR for the new task.
4013 */
4014 if (!(uNewLdt & X86_SEL_MASK_OFF_RPL))
4015 iemHlpLoadNullDataSelectorProt(pVCpu, &pCtx->ldtr, uNewLdt);
4016 else
4017 {
4018 Assert(!pCtx->ldtr.Attr.n.u1Present); /* Ensures that LDT.TI check passes in iemMemFetchSelDesc() below. */
4019
4020 IEMSELDESC DescNewLdt;
4021 rcStrict = iemMemFetchSelDesc(pVCpu, &DescNewLdt, uNewLdt, X86_XCPT_TS);
4022 if (rcStrict != VINF_SUCCESS)
4023 {
4024 Log(("iemTaskSwitch: fetching LDT failed. enmTaskSwitch=%u uNewLdt=%u cbGdt=%u rc=%Rrc\n", enmTaskSwitch,
4025 uNewLdt, pCtx->gdtr.cbGdt, VBOXSTRICTRC_VAL(rcStrict)));
4026 return rcStrict;
4027 }
4028 if ( !DescNewLdt.Legacy.Gen.u1Present
4029 || DescNewLdt.Legacy.Gen.u1DescType
4030 || DescNewLdt.Legacy.Gen.u4Type != X86_SEL_TYPE_SYS_LDT)
4031 {
4032 Log(("iemTaskSwitch: Invalid LDT. enmTaskSwitch=%u uNewLdt=%u DescNewLdt.Legacy.u=%#RX64 -> #TS\n", enmTaskSwitch,
4033 uNewLdt, DescNewLdt.Legacy.u));
4034 return iemRaiseTaskSwitchFaultWithErr(pVCpu, uNewLdt & X86_SEL_MASK_OFF_RPL);
4035 }
4036
4037 pCtx->ldtr.ValidSel = uNewLdt;
4038 pCtx->ldtr.fFlags = CPUMSELREG_FLAGS_VALID;
4039 pCtx->ldtr.u64Base = X86DESC_BASE(&DescNewLdt.Legacy);
4040 pCtx->ldtr.u32Limit = X86DESC_LIMIT_G(&DescNewLdt.Legacy);
4041 pCtx->ldtr.Attr.u = X86DESC_GET_HID_ATTR(&DescNewLdt.Legacy);
4042 if (IEM_IS_GUEST_CPU_INTEL(pVCpu) && !IEM_FULL_VERIFICATION_REM_ENABLED(pVCpu))
4043 pCtx->ldtr.Attr.u &= ~X86DESCATTR_UNUSABLE;
4044 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->ldtr));
4045 }
4046
4047 IEMSELDESC DescSS;
4048 if (IEM_IS_V86_MODE(pVCpu))
4049 {
4050 pVCpu->iem.s.uCpl = 3;
4051 iemHlpLoadSelectorInV86Mode(&pCtx->es, uNewES);
4052 iemHlpLoadSelectorInV86Mode(&pCtx->cs, uNewCS);
4053 iemHlpLoadSelectorInV86Mode(&pCtx->ss, uNewSS);
4054 iemHlpLoadSelectorInV86Mode(&pCtx->ds, uNewDS);
4055 iemHlpLoadSelectorInV86Mode(&pCtx->fs, uNewFS);
4056 iemHlpLoadSelectorInV86Mode(&pCtx->gs, uNewGS);
4057
4058 /* quick fix: fake DescSS. */ /** @todo fix the code further down? */
4059 DescSS.Legacy.u = 0;
4060 DescSS.Legacy.Gen.u16LimitLow = (uint16_t)pCtx->ss.u32Limit;
4061 DescSS.Legacy.Gen.u4LimitHigh = pCtx->ss.u32Limit >> 16;
4062 DescSS.Legacy.Gen.u16BaseLow = (uint16_t)pCtx->ss.u64Base;
4063 DescSS.Legacy.Gen.u8BaseHigh1 = (uint8_t)(pCtx->ss.u64Base >> 16);
4064 DescSS.Legacy.Gen.u8BaseHigh2 = (uint8_t)(pCtx->ss.u64Base >> 24);
4065 DescSS.Legacy.Gen.u4Type = X86_SEL_TYPE_RW_ACC;
4066 DescSS.Legacy.Gen.u2Dpl = 3;
4067 }
4068 else
4069 {
4070 uint8_t uNewCpl = (uNewCS & X86_SEL_RPL);
4071
4072 /*
4073 * Load the stack segment for the new task.
4074 */
4075 if (!(uNewSS & X86_SEL_MASK_OFF_RPL))
4076 {
4077 Log(("iemTaskSwitch: Null stack segment. enmTaskSwitch=%u uNewSS=%#x -> #TS\n", enmTaskSwitch, uNewSS));
4078 return iemRaiseTaskSwitchFaultWithErr(pVCpu, uNewSS & X86_SEL_MASK_OFF_RPL);
4079 }
4080
4081 /* Fetch the descriptor. */
4082 rcStrict = iemMemFetchSelDesc(pVCpu, &DescSS, uNewSS, X86_XCPT_TS);
4083 if (rcStrict != VINF_SUCCESS)
4084 {
4085 Log(("iemTaskSwitch: failed to fetch SS. uNewSS=%#x rc=%Rrc\n", uNewSS,
4086 VBOXSTRICTRC_VAL(rcStrict)));
4087 return rcStrict;
4088 }
4089
4090 /* SS must be a data segment and writable. */
4091 if ( !DescSS.Legacy.Gen.u1DescType
4092 || (DescSS.Legacy.Gen.u4Type & X86_SEL_TYPE_CODE)
4093 || !(DescSS.Legacy.Gen.u4Type & X86_SEL_TYPE_WRITE))
4094 {
4095 Log(("iemTaskSwitch: SS invalid descriptor type. uNewSS=%#x u1DescType=%u u4Type=%#x\n",
4096 uNewSS, DescSS.Legacy.Gen.u1DescType, DescSS.Legacy.Gen.u4Type));
4097 return iemRaiseTaskSwitchFaultWithErr(pVCpu, uNewSS & X86_SEL_MASK_OFF_RPL);
4098 }
4099
4100 /* The SS.RPL, SS.DPL, CS.RPL (CPL) must be equal. */
4101 if ( (uNewSS & X86_SEL_RPL) != uNewCpl
4102 || DescSS.Legacy.Gen.u2Dpl != uNewCpl)
4103 {
4104 Log(("iemTaskSwitch: Invalid priv. for SS. uNewSS=%#x SS.DPL=%u uNewCpl=%u -> #TS\n", uNewSS, DescSS.Legacy.Gen.u2Dpl,
4105 uNewCpl));
4106 return iemRaiseTaskSwitchFaultWithErr(pVCpu, uNewSS & X86_SEL_MASK_OFF_RPL);
4107 }
4108
4109 /* Is it there? */
4110 if (!DescSS.Legacy.Gen.u1Present)
4111 {
4112 Log(("iemTaskSwitch: SS not present. uNewSS=%#x -> #NP\n", uNewSS));
4113 return iemRaiseSelectorNotPresentWithErr(pVCpu, uNewSS & X86_SEL_MASK_OFF_RPL);
4114 }
4115
4116 uint32_t cbLimit = X86DESC_LIMIT_G(&DescSS.Legacy);
4117 uint64_t u64Base = X86DESC_BASE(&DescSS.Legacy);
4118
4119 /* Set the accessed bit before committing the result into SS. */
4120 if (!(DescSS.Legacy.Gen.u4Type & X86_SEL_TYPE_ACCESSED))
4121 {
4122 rcStrict = iemMemMarkSelDescAccessed(pVCpu, uNewSS);
4123 if (rcStrict != VINF_SUCCESS)
4124 return rcStrict;
4125 DescSS.Legacy.Gen.u4Type |= X86_SEL_TYPE_ACCESSED;
4126 }
4127
4128 /* Commit SS. */
4129 pCtx->ss.Sel = uNewSS;
4130 pCtx->ss.ValidSel = uNewSS;
4131 pCtx->ss.Attr.u = X86DESC_GET_HID_ATTR(&DescSS.Legacy);
4132 pCtx->ss.u32Limit = cbLimit;
4133 pCtx->ss.u64Base = u64Base;
4134 pCtx->ss.fFlags = CPUMSELREG_FLAGS_VALID;
4135 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->ss));
4136
4137 /* CPL has changed, update IEM before loading rest of segments. */
4138 pVCpu->iem.s.uCpl = uNewCpl;
4139
4140 /*
4141 * Load the data segments for the new task.
4142 */
4143 rcStrict = iemHlpTaskSwitchLoadDataSelectorInProtMode(pVCpu, &pCtx->es, uNewES);
4144 if (rcStrict != VINF_SUCCESS)
4145 return rcStrict;
4146 rcStrict = iemHlpTaskSwitchLoadDataSelectorInProtMode(pVCpu, &pCtx->ds, uNewDS);
4147 if (rcStrict != VINF_SUCCESS)
4148 return rcStrict;
4149 rcStrict = iemHlpTaskSwitchLoadDataSelectorInProtMode(pVCpu, &pCtx->fs, uNewFS);
4150 if (rcStrict != VINF_SUCCESS)
4151 return rcStrict;
4152 rcStrict = iemHlpTaskSwitchLoadDataSelectorInProtMode(pVCpu, &pCtx->gs, uNewGS);
4153 if (rcStrict != VINF_SUCCESS)
4154 return rcStrict;
4155
4156 /*
4157 * Load the code segment for the new task.
4158 */
4159 if (!(uNewCS & X86_SEL_MASK_OFF_RPL))
4160 {
4161 Log(("iemTaskSwitch #TS: Null code segment. enmTaskSwitch=%u uNewCS=%#x\n", enmTaskSwitch, uNewCS));
4162 return iemRaiseTaskSwitchFaultWithErr(pVCpu, uNewCS & X86_SEL_MASK_OFF_RPL);
4163 }
4164
4165 /* Fetch the descriptor. */
4166 IEMSELDESC DescCS;
4167 rcStrict = iemMemFetchSelDesc(pVCpu, &DescCS, uNewCS, X86_XCPT_TS);
4168 if (rcStrict != VINF_SUCCESS)
4169 {
4170 Log(("iemTaskSwitch: failed to fetch CS. uNewCS=%u rc=%Rrc\n", uNewCS, VBOXSTRICTRC_VAL(rcStrict)));
4171 return rcStrict;
4172 }
4173
4174 /* CS must be a code segment. */
4175 if ( !DescCS.Legacy.Gen.u1DescType
4176 || !(DescCS.Legacy.Gen.u4Type & X86_SEL_TYPE_CODE))
4177 {
4178 Log(("iemTaskSwitch: CS invalid descriptor type. uNewCS=%#x u1DescType=%u u4Type=%#x -> #TS\n", uNewCS,
4179 DescCS.Legacy.Gen.u1DescType, DescCS.Legacy.Gen.u4Type));
4180 return iemRaiseTaskSwitchFaultWithErr(pVCpu, uNewCS & X86_SEL_MASK_OFF_RPL);
4181 }
4182
4183 /* For conforming CS, DPL must be less than or equal to the RPL. */
4184 if ( (DescCS.Legacy.Gen.u4Type & X86_SEL_TYPE_CONF)
4185 && DescCS.Legacy.Gen.u2Dpl > (uNewCS & X86_SEL_RPL))
4186 {
4187 Log(("iemTaskSwitch: confirming CS DPL > RPL. uNewCS=%#x u4Type=%#x DPL=%u -> #TS\n", uNewCS, DescCS.Legacy.Gen.u4Type,
4188 DescCS.Legacy.Gen.u2Dpl));
4189 return iemRaiseTaskSwitchFaultWithErr(pVCpu, uNewCS & X86_SEL_MASK_OFF_RPL);
4190 }
4191
4192 /* For non-conforming CS, DPL must match RPL. */
4193 if ( !(DescCS.Legacy.Gen.u4Type & X86_SEL_TYPE_CONF)
4194 && DescCS.Legacy.Gen.u2Dpl != (uNewCS & X86_SEL_RPL))
4195 {
4196 Log(("iemTaskSwitch: non-confirming CS DPL RPL mismatch. uNewCS=%#x u4Type=%#x DPL=%u -> #TS\n", uNewCS,
4197 DescCS.Legacy.Gen.u4Type, DescCS.Legacy.Gen.u2Dpl));
4198 return iemRaiseTaskSwitchFaultWithErr(pVCpu, uNewCS & X86_SEL_MASK_OFF_RPL);
4199 }
4200
4201 /* Is it there? */
4202 if (!DescCS.Legacy.Gen.u1Present)
4203 {
4204 Log(("iemTaskSwitch: CS not present. uNewCS=%#x -> #NP\n", uNewCS));
4205 return iemRaiseSelectorNotPresentWithErr(pVCpu, uNewCS & X86_SEL_MASK_OFF_RPL);
4206 }
4207
4208 cbLimit = X86DESC_LIMIT_G(&DescCS.Legacy);
4209 u64Base = X86DESC_BASE(&DescCS.Legacy);
4210
4211 /* Set the accessed bit before committing the result into CS. */
4212 if (!(DescCS.Legacy.Gen.u4Type & X86_SEL_TYPE_ACCESSED))
4213 {
4214 rcStrict = iemMemMarkSelDescAccessed(pVCpu, uNewCS);
4215 if (rcStrict != VINF_SUCCESS)
4216 return rcStrict;
4217 DescCS.Legacy.Gen.u4Type |= X86_SEL_TYPE_ACCESSED;
4218 }
4219
4220 /* Commit CS. */
4221 pCtx->cs.Sel = uNewCS;
4222 pCtx->cs.ValidSel = uNewCS;
4223 pCtx->cs.Attr.u = X86DESC_GET_HID_ATTR(&DescCS.Legacy);
4224 pCtx->cs.u32Limit = cbLimit;
4225 pCtx->cs.u64Base = u64Base;
4226 pCtx->cs.fFlags = CPUMSELREG_FLAGS_VALID;
4227 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &pCtx->cs));
4228 }
4229
4230 /** @todo Debug trap. */
4231 if (fIsNewTSS386 && fNewDebugTrap)
4232 Log(("iemTaskSwitch: Debug Trap set in new TSS. Not implemented!\n"));
4233
4234 /*
4235 * Construct the error code masks based on what caused this task switch.
4236 * See Intel Instruction reference for INT.
4237 */
4238 uint16_t uExt;
4239 if ( enmTaskSwitch == IEMTASKSWITCH_INT_XCPT
4240 && !(fFlags & IEM_XCPT_FLAGS_T_SOFT_INT))
4241 {
4242 uExt = 1;
4243 }
4244 else
4245 uExt = 0;
4246
4247 /*
4248 * Push any error code on to the new stack.
4249 */
4250 if (fFlags & IEM_XCPT_FLAGS_ERR)
4251 {
4252 Assert(enmTaskSwitch == IEMTASKSWITCH_INT_XCPT);
4253 uint32_t cbLimitSS = X86DESC_LIMIT_G(&DescSS.Legacy);
4254 uint8_t const cbStackFrame = fIsNewTSS386 ? 4 : 2;
4255
4256 /* Check that there is sufficient space on the stack. */
4257 /** @todo Factor out segment limit checking for normal/expand down segments
4258 * into a separate function. */
4259 if (!(DescSS.Legacy.Gen.u4Type & X86_SEL_TYPE_DOWN))
4260 {
4261 if ( pCtx->esp - 1 > cbLimitSS
4262 || pCtx->esp < cbStackFrame)
4263 {
4264 /** @todo Intel says \#SS(EXT) for INT/XCPT, I couldn't figure out AMD yet. */
4265 Log(("iemTaskSwitch: SS=%#x ESP=%#x cbStackFrame=%#x is out of bounds -> #SS\n",
4266 pCtx->ss.Sel, pCtx->esp, cbStackFrame));
4267 return iemRaiseStackSelectorNotPresentWithErr(pVCpu, uExt);
4268 }
4269 }
4270 else
4271 {
4272 if ( pCtx->esp - 1 > (DescSS.Legacy.Gen.u1DefBig ? UINT32_MAX : UINT32_C(0xffff))
4273 || pCtx->esp - cbStackFrame < cbLimitSS + UINT32_C(1))
4274 {
4275 Log(("iemTaskSwitch: SS=%#x ESP=%#x cbStackFrame=%#x (expand down) is out of bounds -> #SS\n",
4276 pCtx->ss.Sel, pCtx->esp, cbStackFrame));
4277 return iemRaiseStackSelectorNotPresentWithErr(pVCpu, uExt);
4278 }
4279 }
4280
4281
4282 if (fIsNewTSS386)
4283 rcStrict = iemMemStackPushU32(pVCpu, uErr);
4284 else
4285 rcStrict = iemMemStackPushU16(pVCpu, uErr);
4286 if (rcStrict != VINF_SUCCESS)
4287 {
4288 Log(("iemTaskSwitch: Can't push error code to new task's stack. %s-bit TSS. rc=%Rrc\n",
4289 fIsNewTSS386 ? "32" : "16", VBOXSTRICTRC_VAL(rcStrict)));
4290 return rcStrict;
4291 }
4292 }
4293
4294 /* Check the new EIP against the new CS limit. */
4295 if (pCtx->eip > pCtx->cs.u32Limit)
4296 {
4297 Log(("iemHlpTaskSwitchLoadDataSelectorInProtMode: New EIP exceeds CS limit. uNewEIP=%#RX32 CS limit=%u -> #GP(0)\n",
4298 pCtx->eip, pCtx->cs.u32Limit));
4299 /** @todo Intel says \#GP(EXT) for INT/XCPT, I couldn't figure out AMD yet. */
4300 return iemRaiseGeneralProtectionFault(pVCpu, uExt);
4301 }
4302
4303 Log(("iemTaskSwitch: Success! New CS:EIP=%#04x:%#x SS=%#04x\n", pCtx->cs.Sel, pCtx->eip, pCtx->ss.Sel));
4304 return fFlags & IEM_XCPT_FLAGS_T_CPU_XCPT ? VINF_IEM_RAISED_XCPT : VINF_SUCCESS;
4305}
4306
4307
4308/**
4309 * Implements exceptions and interrupts for protected mode.
4310 *
4311 * @returns VBox strict status code.
4312 * @param pVCpu The cross context virtual CPU structure of the calling thread.
4313 * @param pCtx The CPU context.
4314 * @param cbInstr The number of bytes to offset rIP by in the return
4315 * address.
4316 * @param u8Vector The interrupt / exception vector number.
4317 * @param fFlags The flags.
4318 * @param uErr The error value if IEM_XCPT_FLAGS_ERR is set.
4319 * @param uCr2 The CR2 value if IEM_XCPT_FLAGS_CR2 is set.
4320 */
4321IEM_STATIC VBOXSTRICTRC
4322iemRaiseXcptOrIntInProtMode(PVMCPU pVCpu,
4323 PCPUMCTX pCtx,
4324 uint8_t cbInstr,
4325 uint8_t u8Vector,
4326 uint32_t fFlags,
4327 uint16_t uErr,
4328 uint64_t uCr2)
4329{
4330 /*
4331 * Read the IDT entry.
4332 */
4333 if (pCtx->idtr.cbIdt < UINT32_C(8) * u8Vector + 7)
4334 {
4335 Log(("RaiseXcptOrIntInProtMode: %#x is out of bounds (%#x)\n", u8Vector, pCtx->idtr.cbIdt));
4336 return iemRaiseGeneralProtectionFault(pVCpu, X86_TRAP_ERR_IDT | ((uint16_t)u8Vector << X86_TRAP_ERR_SEL_SHIFT));
4337 }
4338 X86DESC Idte;
4339 VBOXSTRICTRC rcStrict = iemMemFetchSysU64(pVCpu, &Idte.u, UINT8_MAX,
4340 pCtx->idtr.pIdt + UINT32_C(8) * u8Vector);
4341 if (RT_UNLIKELY(rcStrict != VINF_SUCCESS))
4342 return rcStrict;
4343 Log(("iemRaiseXcptOrIntInProtMode: vec=%#x P=%u DPL=%u DT=%u:%u A=%u %04x:%04x%04x\n",
4344 u8Vector, Idte.Gate.u1Present, Idte.Gate.u2Dpl, Idte.Gate.u1DescType, Idte.Gate.u4Type,
4345 Idte.Gate.u4ParmCount, Idte.Gate.u16Sel, Idte.Gate.u16OffsetHigh, Idte.Gate.u16OffsetLow));
4346
4347 /*
4348 * Check the descriptor type, DPL and such.
4349 * ASSUMES this is done in the same order as described for call-gate calls.
4350 */
4351 if (Idte.Gate.u1DescType)
4352 {
4353 Log(("RaiseXcptOrIntInProtMode %#x - not system selector (%#x) -> #GP\n", u8Vector, Idte.Gate.u4Type));
4354 return iemRaiseGeneralProtectionFault(pVCpu, X86_TRAP_ERR_IDT | ((uint16_t)u8Vector << X86_TRAP_ERR_SEL_SHIFT));
4355 }
4356 bool fTaskGate = false;
4357 uint8_t f32BitGate = true;
4358 uint32_t fEflToClear = X86_EFL_TF | X86_EFL_NT | X86_EFL_RF | X86_EFL_VM;
4359 switch (Idte.Gate.u4Type)
4360 {
4361 case X86_SEL_TYPE_SYS_UNDEFINED:
4362 case X86_SEL_TYPE_SYS_286_TSS_AVAIL:
4363 case X86_SEL_TYPE_SYS_LDT:
4364 case X86_SEL_TYPE_SYS_286_TSS_BUSY:
4365 case X86_SEL_TYPE_SYS_286_CALL_GATE:
4366 case X86_SEL_TYPE_SYS_UNDEFINED2:
4367 case X86_SEL_TYPE_SYS_386_TSS_AVAIL:
4368 case X86_SEL_TYPE_SYS_UNDEFINED3:
4369 case X86_SEL_TYPE_SYS_386_TSS_BUSY:
4370 case X86_SEL_TYPE_SYS_386_CALL_GATE:
4371 case X86_SEL_TYPE_SYS_UNDEFINED4:
4372 {
4373 /** @todo check what actually happens when the type is wrong...
4374 * esp. call gates. */
4375 Log(("RaiseXcptOrIntInProtMode %#x - invalid type (%#x) -> #GP\n", u8Vector, Idte.Gate.u4Type));
4376 return iemRaiseGeneralProtectionFault(pVCpu, X86_TRAP_ERR_IDT | ((uint16_t)u8Vector << X86_TRAP_ERR_SEL_SHIFT));
4377 }
4378
4379 case X86_SEL_TYPE_SYS_286_INT_GATE:
4380 f32BitGate = false;
4381 case X86_SEL_TYPE_SYS_386_INT_GATE:
4382 fEflToClear |= X86_EFL_IF;
4383 break;
4384
4385 case X86_SEL_TYPE_SYS_TASK_GATE:
4386 fTaskGate = true;
4387#ifndef IEM_IMPLEMENTS_TASKSWITCH
4388 IEM_RETURN_ASPECT_NOT_IMPLEMENTED_LOG(("Task gates\n"));
4389#endif
4390 break;
4391
4392 case X86_SEL_TYPE_SYS_286_TRAP_GATE:
4393 f32BitGate = false;
4394 case X86_SEL_TYPE_SYS_386_TRAP_GATE:
4395 break;
4396
4397 IEM_NOT_REACHED_DEFAULT_CASE_RET();
4398 }
4399
4400 /* Check DPL against CPL if applicable. */
4401 if (fFlags & IEM_XCPT_FLAGS_T_SOFT_INT)
4402 {
4403 if (pVCpu->iem.s.uCpl > Idte.Gate.u2Dpl)
4404 {
4405 Log(("RaiseXcptOrIntInProtMode %#x - CPL (%d) > DPL (%d) -> #GP\n", u8Vector, pVCpu->iem.s.uCpl, Idte.Gate.u2Dpl));
4406 return iemRaiseGeneralProtectionFault(pVCpu, X86_TRAP_ERR_IDT | ((uint16_t)u8Vector << X86_TRAP_ERR_SEL_SHIFT));
4407 }
4408 }
4409
4410 /* Is it there? */
4411 if (!Idte.Gate.u1Present)
4412 {
4413 Log(("RaiseXcptOrIntInProtMode %#x - not present -> #NP\n", u8Vector));
4414 return iemRaiseSelectorNotPresentWithErr(pVCpu, X86_TRAP_ERR_IDT | ((uint16_t)u8Vector << X86_TRAP_ERR_SEL_SHIFT));
4415 }
4416
4417 /* Is it a task-gate? */
4418 if (fTaskGate)
4419 {
4420 /*
4421 * Construct the error code masks based on what caused this task switch.
4422 * See Intel Instruction reference for INT.
4423 */
4424 uint16_t const uExt = (fFlags & IEM_XCPT_FLAGS_T_SOFT_INT) ? 0 : 1;
4425 uint16_t const uSelMask = X86_SEL_MASK_OFF_RPL;
4426 RTSEL SelTSS = Idte.Gate.u16Sel;
4427
4428 /*
4429 * Fetch the TSS descriptor in the GDT.
4430 */
4431 IEMSELDESC DescTSS;
4432 rcStrict = iemMemFetchSelDescWithErr(pVCpu, &DescTSS, SelTSS, X86_XCPT_GP, (SelTSS & uSelMask) | uExt);
4433 if (rcStrict != VINF_SUCCESS)
4434 {
4435 Log(("RaiseXcptOrIntInProtMode %#x - failed to fetch TSS selector %#x, rc=%Rrc\n", u8Vector, SelTSS,
4436 VBOXSTRICTRC_VAL(rcStrict)));
4437 return rcStrict;
4438 }
4439
4440 /* The TSS descriptor must be a system segment and be available (not busy). */
4441 if ( DescTSS.Legacy.Gen.u1DescType
4442 || ( DescTSS.Legacy.Gen.u4Type != X86_SEL_TYPE_SYS_286_TSS_AVAIL
4443 && DescTSS.Legacy.Gen.u4Type != X86_SEL_TYPE_SYS_386_TSS_AVAIL))
4444 {
4445 Log(("RaiseXcptOrIntInProtMode %#x - TSS selector %#x of task gate not a system descriptor or not available %#RX64\n",
4446 u8Vector, SelTSS, DescTSS.Legacy.au64));
4447 return iemRaiseGeneralProtectionFault(pVCpu, (SelTSS & uSelMask) | uExt);
4448 }
4449
4450 /* The TSS must be present. */
4451 if (!DescTSS.Legacy.Gen.u1Present)
4452 {
4453 Log(("RaiseXcptOrIntInProtMode %#x - TSS selector %#x not present %#RX64\n", u8Vector, SelTSS, DescTSS.Legacy.au64));
4454 return iemRaiseSelectorNotPresentWithErr(pVCpu, (SelTSS & uSelMask) | uExt);
4455 }
4456
4457 /* Do the actual task switch. */
4458 return iemTaskSwitch(pVCpu, pCtx, IEMTASKSWITCH_INT_XCPT, pCtx->eip, fFlags, uErr, uCr2, SelTSS, &DescTSS);
4459 }
4460
4461 /* A null CS is bad. */
4462 RTSEL NewCS = Idte.Gate.u16Sel;
4463 if (!(NewCS & X86_SEL_MASK_OFF_RPL))
4464 {
4465 Log(("RaiseXcptOrIntInProtMode %#x - CS=%#x -> #GP\n", u8Vector, NewCS));
4466 return iemRaiseGeneralProtectionFault0(pVCpu);
4467 }
4468
4469 /* Fetch the descriptor for the new CS. */
4470 IEMSELDESC DescCS;
4471 rcStrict = iemMemFetchSelDesc(pVCpu, &DescCS, NewCS, X86_XCPT_GP); /** @todo correct exception? */
4472 if (rcStrict != VINF_SUCCESS)
4473 {
4474 Log(("RaiseXcptOrIntInProtMode %#x - CS=%#x - rc=%Rrc\n", u8Vector, NewCS, VBOXSTRICTRC_VAL(rcStrict)));
4475 return rcStrict;
4476 }
4477
4478 /* Must be a code segment. */
4479 if (!DescCS.Legacy.Gen.u1DescType)
4480 {
4481 Log(("RaiseXcptOrIntInProtMode %#x - CS=%#x - system selector (%#x) -> #GP\n", u8Vector, NewCS, DescCS.Legacy.Gen.u4Type));
4482 return iemRaiseGeneralProtectionFault(pVCpu, NewCS & X86_SEL_MASK_OFF_RPL);
4483 }
4484 if (!(DescCS.Legacy.Gen.u4Type & X86_SEL_TYPE_CODE))
4485 {
4486 Log(("RaiseXcptOrIntInProtMode %#x - CS=%#x - data selector (%#x) -> #GP\n", u8Vector, NewCS, DescCS.Legacy.Gen.u4Type));
4487 return iemRaiseGeneralProtectionFault(pVCpu, NewCS & X86_SEL_MASK_OFF_RPL);
4488 }
4489
4490 /* Don't allow lowering the privilege level. */
4491 /** @todo Does the lowering of privileges apply to software interrupts
4492 * only? This has bearings on the more-privileged or
4493 * same-privilege stack behavior further down. A testcase would
4494 * be nice. */
4495 if (DescCS.Legacy.Gen.u2Dpl > pVCpu->iem.s.uCpl)
4496 {
4497 Log(("RaiseXcptOrIntInProtMode %#x - CS=%#x - DPL (%d) > CPL (%d) -> #GP\n",
4498 u8Vector, NewCS, DescCS.Legacy.Gen.u2Dpl, pVCpu->iem.s.uCpl));
4499 return iemRaiseGeneralProtectionFault(pVCpu, NewCS & X86_SEL_MASK_OFF_RPL);
4500 }
4501
4502 /* Make sure the selector is present. */
4503 if (!DescCS.Legacy.Gen.u1Present)
4504 {
4505 Log(("RaiseXcptOrIntInProtMode %#x - CS=%#x - segment not present -> #NP\n", u8Vector, NewCS));
4506 return iemRaiseSelectorNotPresentBySelector(pVCpu, NewCS);
4507 }
4508
4509 /* Check the new EIP against the new CS limit. */
4510 uint32_t const uNewEip = Idte.Gate.u4Type == X86_SEL_TYPE_SYS_286_INT_GATE
4511 || Idte.Gate.u4Type == X86_SEL_TYPE_SYS_286_TRAP_GATE
4512 ? Idte.Gate.u16OffsetLow
4513 : Idte.Gate.u16OffsetLow | ((uint32_t)Idte.Gate.u16OffsetHigh << 16);
4514 uint32_t cbLimitCS = X86DESC_LIMIT_G(&DescCS.Legacy);
4515 if (uNewEip > cbLimitCS)
4516 {
4517 Log(("RaiseXcptOrIntInProtMode %#x - EIP=%#x > cbLimitCS=%#x (CS=%#x) -> #GP(0)\n",
4518 u8Vector, uNewEip, cbLimitCS, NewCS));
4519 return iemRaiseGeneralProtectionFault(pVCpu, 0);
4520 }
4521
4522 /* Calc the flag image to push. */
4523 uint32_t fEfl = IEMMISC_GET_EFL(pVCpu, pCtx);
4524 if (fFlags & (IEM_XCPT_FLAGS_DRx_INSTR_BP | IEM_XCPT_FLAGS_T_SOFT_INT))
4525 fEfl &= ~X86_EFL_RF;
4526 else if (!IEM_FULL_VERIFICATION_REM_ENABLED(pVCpu))
4527 fEfl |= X86_EFL_RF; /* Vagueness is all I've found on this so far... */ /** @todo Automatically pushing EFLAGS.RF. */
4528
4529 /* From V8086 mode only go to CPL 0. */
4530 uint8_t const uNewCpl = DescCS.Legacy.Gen.u4Type & X86_SEL_TYPE_CONF
4531 ? pVCpu->iem.s.uCpl : DescCS.Legacy.Gen.u2Dpl;
4532 if ((fEfl & X86_EFL_VM) && uNewCpl != 0) /** @todo When exactly is this raised? */
4533 {
4534 Log(("RaiseXcptOrIntInProtMode %#x - CS=%#x - New CPL (%d) != 0 w/ VM=1 -> #GP\n", u8Vector, NewCS, uNewCpl));
4535 return iemRaiseGeneralProtectionFault(pVCpu, 0);
4536 }
4537
4538 /*
4539 * If the privilege level changes, we need to get a new stack from the TSS.
4540 * This in turns means validating the new SS and ESP...
4541 */
4542 if (uNewCpl != pVCpu->iem.s.uCpl)
4543 {
4544 RTSEL NewSS;
4545 uint32_t uNewEsp;
4546 rcStrict = iemRaiseLoadStackFromTss32Or16(pVCpu, pCtx, uNewCpl, &NewSS, &uNewEsp);
4547 if (rcStrict != VINF_SUCCESS)
4548 return rcStrict;
4549
4550 IEMSELDESC DescSS;
4551 rcStrict = iemMiscValidateNewSS(pVCpu, pCtx, NewSS, uNewCpl, &DescSS);
4552 if (rcStrict != VINF_SUCCESS)
4553 return rcStrict;
4554
4555 /* Check that there is sufficient space for the stack frame. */
4556 uint32_t cbLimitSS = X86DESC_LIMIT_G(&DescSS.Legacy);
4557 uint8_t const cbStackFrame = !(fEfl & X86_EFL_VM)
4558 ? (fFlags & IEM_XCPT_FLAGS_ERR ? 12 : 10) << f32BitGate
4559 : (fFlags & IEM_XCPT_FLAGS_ERR ? 20 : 18) << f32BitGate;
4560
4561 if (!(DescSS.Legacy.Gen.u4Type & X86_SEL_TYPE_DOWN))
4562 {
4563 if ( uNewEsp - 1 > cbLimitSS
4564 || uNewEsp < cbStackFrame)
4565 {
4566 Log(("RaiseXcptOrIntInProtMode: %#x - SS=%#x ESP=%#x cbStackFrame=%#x is out of bounds -> #GP\n",
4567 u8Vector, NewSS, uNewEsp, cbStackFrame));
4568 return iemRaiseSelectorBoundsBySelector(pVCpu, NewSS);
4569 }
4570 }
4571 else
4572 {
4573 if ( uNewEsp - 1 > (DescSS.Legacy.Gen.u4Type & X86_DESC_DB ? UINT32_MAX : UINT32_C(0xffff))
4574 || uNewEsp - cbStackFrame < cbLimitSS + UINT32_C(1))
4575 {
4576 Log(("RaiseXcptOrIntInProtMode: %#x - SS=%#x ESP=%#x cbStackFrame=%#x (expand down) is out of bounds -> #GP\n",
4577 u8Vector, NewSS, uNewEsp, cbStackFrame));
4578 return iemRaiseSelectorBoundsBySelector(pVCpu, NewSS);
4579 }
4580 }
4581
4582 /*
4583 * Start making changes.
4584 */
4585
4586 /* Set the new CPL so that stack accesses use it. */
4587 uint8_t const uOldCpl = pVCpu->iem.s.uCpl;
4588 pVCpu->iem.s.uCpl = uNewCpl;
4589
4590 /* Create the stack frame. */
4591 RTPTRUNION uStackFrame;
4592 rcStrict = iemMemMap(pVCpu, &uStackFrame.pv, cbStackFrame, UINT8_MAX,
4593 uNewEsp - cbStackFrame + X86DESC_BASE(&DescSS.Legacy), IEM_ACCESS_STACK_W | IEM_ACCESS_WHAT_SYS); /* _SYS is a hack ... */
4594 if (rcStrict != VINF_SUCCESS)
4595 return rcStrict;
4596 void * const pvStackFrame = uStackFrame.pv;
4597 if (f32BitGate)
4598 {
4599 if (fFlags & IEM_XCPT_FLAGS_ERR)
4600 *uStackFrame.pu32++ = uErr;
4601 uStackFrame.pu32[0] = (fFlags & IEM_XCPT_FLAGS_T_SOFT_INT) ? pCtx->eip + cbInstr : pCtx->eip;
4602 uStackFrame.pu32[1] = (pCtx->cs.Sel & ~X86_SEL_RPL) | uOldCpl;
4603 uStackFrame.pu32[2] = fEfl;
4604 uStackFrame.pu32[3] = pCtx->esp;
4605 uStackFrame.pu32[4] = pCtx->ss.Sel;
4606 if (fEfl & X86_EFL_VM)
4607 {
4608 uStackFrame.pu32[1] = pCtx->cs.Sel;
4609 uStackFrame.pu32[5] = pCtx->es.Sel;
4610 uStackFrame.pu32[6] = pCtx->ds.Sel;
4611 uStackFrame.pu32[7] = pCtx->fs.Sel;
4612 uStackFrame.pu32[8] = pCtx->gs.Sel;
4613 }
4614 }
4615 else
4616 {
4617 if (fFlags & IEM_XCPT_FLAGS_ERR)
4618 *uStackFrame.pu16++ = uErr;
4619 uStackFrame.pu16[0] = (fFlags & IEM_XCPT_FLAGS_T_SOFT_INT) ? pCtx->ip + cbInstr : pCtx->ip;
4620 uStackFrame.pu16[1] = (pCtx->cs.Sel & ~X86_SEL_RPL) | uOldCpl;
4621 uStackFrame.pu16[2] = fEfl;
4622 uStackFrame.pu16[3] = pCtx->sp;
4623 uStackFrame.pu16[4] = pCtx->ss.Sel;
4624 if (fEfl & X86_EFL_VM)
4625 {
4626 uStackFrame.pu16[1] = pCtx->cs.Sel;
4627 uStackFrame.pu16[5] = pCtx->es.Sel;
4628 uStackFrame.pu16[6] = pCtx->ds.Sel;
4629 uStackFrame.pu16[7] = pCtx->fs.Sel;
4630 uStackFrame.pu16[8] = pCtx->gs.Sel;
4631 }
4632 }
4633 rcStrict = iemMemCommitAndUnmap(pVCpu, pvStackFrame, IEM_ACCESS_STACK_W | IEM_ACCESS_WHAT_SYS);
4634 if (rcStrict != VINF_SUCCESS)
4635 return rcStrict;
4636
4637 /* Mark the selectors 'accessed' (hope this is the correct time). */
4638 /** @todo testcase: excatly _when_ are the accessed bits set - before or
4639 * after pushing the stack frame? (Write protect the gdt + stack to
4640 * find out.) */
4641 if (!(DescCS.Legacy.Gen.u4Type & X86_SEL_TYPE_ACCESSED))
4642 {
4643 rcStrict = iemMemMarkSelDescAccessed(pVCpu, NewCS);
4644 if (rcStrict != VINF_SUCCESS)
4645 return rcStrict;
4646 DescCS.Legacy.Gen.u4Type |= X86_SEL_TYPE_ACCESSED;
4647 }
4648
4649 if (!(DescSS.Legacy.Gen.u4Type & X86_SEL_TYPE_ACCESSED))
4650 {
4651 rcStrict = iemMemMarkSelDescAccessed(pVCpu, NewSS);
4652 if (rcStrict != VINF_SUCCESS)
4653 return rcStrict;
4654 DescSS.Legacy.Gen.u4Type |= X86_SEL_TYPE_ACCESSED;
4655 }
4656
4657 /*
4658 * Start comitting the register changes (joins with the DPL=CPL branch).
4659 */
4660 pCtx->ss.Sel = NewSS;
4661 pCtx->ss.ValidSel = NewSS;
4662 pCtx->ss.fFlags = CPUMSELREG_FLAGS_VALID;
4663 pCtx->ss.u32Limit = cbLimitSS;
4664 pCtx->ss.u64Base = X86DESC_BASE(&DescSS.Legacy);
4665 pCtx->ss.Attr.u = X86DESC_GET_HID_ATTR(&DescSS.Legacy);
4666 /** @todo When coming from 32-bit code and operating with a 16-bit TSS and
4667 * 16-bit handler, the high word of ESP remains unchanged (i.e. only
4668 * SP is loaded).
4669 * Need to check the other combinations too:
4670 * - 16-bit TSS, 32-bit handler
4671 * - 32-bit TSS, 16-bit handler */
4672 if (!pCtx->ss.Attr.n.u1DefBig)
4673 pCtx->sp = (uint16_t)(uNewEsp - cbStackFrame);
4674 else
4675 pCtx->rsp = uNewEsp - cbStackFrame;
4676
4677 if (fEfl & X86_EFL_VM)
4678 {
4679 iemHlpLoadNullDataSelectorOnV86Xcpt(pVCpu, &pCtx->gs);
4680 iemHlpLoadNullDataSelectorOnV86Xcpt(pVCpu, &pCtx->fs);
4681 iemHlpLoadNullDataSelectorOnV86Xcpt(pVCpu, &pCtx->es);
4682 iemHlpLoadNullDataSelectorOnV86Xcpt(pVCpu, &pCtx->ds);
4683 }
4684 }
4685 /*
4686 * Same privilege, no stack change and smaller stack frame.
4687 */
4688 else
4689 {
4690 uint64_t uNewRsp;
4691 RTPTRUNION uStackFrame;
4692 uint8_t const cbStackFrame = (fFlags & IEM_XCPT_FLAGS_ERR ? 8 : 6) << f32BitGate;
4693 rcStrict = iemMemStackPushBeginSpecial(pVCpu, cbStackFrame, &uStackFrame.pv, &uNewRsp);
4694 if (rcStrict != VINF_SUCCESS)
4695 return rcStrict;
4696 void * const pvStackFrame = uStackFrame.pv;
4697
4698 if (f32BitGate)
4699 {
4700 if (fFlags & IEM_XCPT_FLAGS_ERR)
4701 *uStackFrame.pu32++ = uErr;
4702 uStackFrame.pu32[0] = fFlags & IEM_XCPT_FLAGS_T_SOFT_INT ? pCtx->eip + cbInstr : pCtx->eip;
4703 uStackFrame.pu32[1] = (pCtx->cs.Sel & ~X86_SEL_RPL) | pVCpu->iem.s.uCpl;
4704 uStackFrame.pu32[2] = fEfl;
4705 }
4706 else
4707 {
4708 if (fFlags & IEM_XCPT_FLAGS_ERR)
4709 *uStackFrame.pu16++ = uErr;
4710 uStackFrame.pu16[0] = fFlags & IEM_XCPT_FLAGS_T_SOFT_INT ? pCtx->eip + cbInstr : pCtx->eip;
4711 uStackFrame.pu16[1] = (pCtx->cs.Sel & ~X86_SEL_RPL) | pVCpu->iem.s.uCpl;
4712 uStackFrame.pu16[2] = fEfl;
4713 }
4714 rcStrict = iemMemCommitAndUnmap(pVCpu, pvStackFrame, IEM_ACCESS_STACK_W); /* don't use the commit here */
4715 if (rcStrict != VINF_SUCCESS)
4716 return rcStrict;
4717
4718 /* Mark the CS selector as 'accessed'. */
4719 if (!(DescCS.Legacy.Gen.u4Type & X86_SEL_TYPE_ACCESSED))
4720 {
4721 rcStrict = iemMemMarkSelDescAccessed(pVCpu, NewCS);
4722 if (rcStrict != VINF_SUCCESS)
4723 return rcStrict;
4724 DescCS.Legacy.Gen.u4Type |= X86_SEL_TYPE_ACCESSED;
4725 }
4726
4727 /*
4728 * Start committing the register changes (joins with the other branch).
4729 */
4730 pCtx->rsp = uNewRsp;
4731 }
4732
4733 /* ... register committing continues. */
4734 pCtx->cs.Sel = (NewCS & ~X86_SEL_RPL) | uNewCpl;
4735 pCtx->cs.ValidSel = (NewCS & ~X86_SEL_RPL) | uNewCpl;
4736 pCtx->cs.fFlags = CPUMSELREG_FLAGS_VALID;
4737 pCtx->cs.u32Limit = cbLimitCS;
4738 pCtx->cs.u64Base = X86DESC_BASE(&DescCS.Legacy);
4739 pCtx->cs.Attr.u = X86DESC_GET_HID_ATTR(&DescCS.Legacy);
4740
4741 pCtx->rip = uNewEip; /* (The entire register is modified, see pe16_32 bs3kit tests.) */
4742 fEfl &= ~fEflToClear;
4743 IEMMISC_SET_EFL(pVCpu, pCtx, fEfl);
4744
4745 if (fFlags & IEM_XCPT_FLAGS_CR2)
4746 pCtx->cr2 = uCr2;
4747
4748 if (fFlags & IEM_XCPT_FLAGS_T_CPU_XCPT)
4749 iemRaiseXcptAdjustState(pCtx, u8Vector);
4750
4751 return fFlags & IEM_XCPT_FLAGS_T_CPU_XCPT ? VINF_IEM_RAISED_XCPT : VINF_SUCCESS;
4752}
4753
4754
4755/**
4756 * Implements exceptions and interrupts for long mode.
4757 *
4758 * @returns VBox strict status code.
4759 * @param pVCpu The cross context virtual CPU structure of the calling thread.
4760 * @param pCtx The CPU context.
4761 * @param cbInstr The number of bytes to offset rIP by in the return
4762 * address.
4763 * @param u8Vector The interrupt / exception vector number.
4764 * @param fFlags The flags.
4765 * @param uErr The error value if IEM_XCPT_FLAGS_ERR is set.
4766 * @param uCr2 The CR2 value if IEM_XCPT_FLAGS_CR2 is set.
4767 */
4768IEM_STATIC VBOXSTRICTRC
4769iemRaiseXcptOrIntInLongMode(PVMCPU pVCpu,
4770 PCPUMCTX pCtx,
4771 uint8_t cbInstr,
4772 uint8_t u8Vector,
4773 uint32_t fFlags,
4774 uint16_t uErr,
4775 uint64_t uCr2)
4776{
4777 /*
4778 * Read the IDT entry.
4779 */
4780 uint16_t offIdt = (uint16_t)u8Vector << 4;
4781 if (pCtx->idtr.cbIdt < offIdt + 7)
4782 {
4783 Log(("iemRaiseXcptOrIntInLongMode: %#x is out of bounds (%#x)\n", u8Vector, pCtx->idtr.cbIdt));
4784 return iemRaiseGeneralProtectionFault(pVCpu, X86_TRAP_ERR_IDT | ((uint16_t)u8Vector << X86_TRAP_ERR_SEL_SHIFT));
4785 }
4786 X86DESC64 Idte;
4787 VBOXSTRICTRC rcStrict = iemMemFetchSysU64(pVCpu, &Idte.au64[0], UINT8_MAX, pCtx->idtr.pIdt + offIdt);
4788 if (RT_LIKELY(rcStrict == VINF_SUCCESS))
4789 rcStrict = iemMemFetchSysU64(pVCpu, &Idte.au64[1], UINT8_MAX, pCtx->idtr.pIdt + offIdt + 8);
4790 if (RT_UNLIKELY(rcStrict != VINF_SUCCESS))
4791 return rcStrict;
4792 Log(("iemRaiseXcptOrIntInLongMode: vec=%#x P=%u DPL=%u DT=%u:%u IST=%u %04x:%08x%04x%04x\n",
4793 u8Vector, Idte.Gate.u1Present, Idte.Gate.u2Dpl, Idte.Gate.u1DescType, Idte.Gate.u4Type,
4794 Idte.Gate.u3IST, Idte.Gate.u16Sel, Idte.Gate.u32OffsetTop, Idte.Gate.u16OffsetHigh, Idte.Gate.u16OffsetLow));
4795
4796 /*
4797 * Check the descriptor type, DPL and such.
4798 * ASSUMES this is done in the same order as described for call-gate calls.
4799 */
4800 if (Idte.Gate.u1DescType)
4801 {
4802 Log(("iemRaiseXcptOrIntInLongMode %#x - not system selector (%#x) -> #GP\n", u8Vector, Idte.Gate.u4Type));
4803 return iemRaiseGeneralProtectionFault(pVCpu, X86_TRAP_ERR_IDT | ((uint16_t)u8Vector << X86_TRAP_ERR_SEL_SHIFT));
4804 }
4805 uint32_t fEflToClear = X86_EFL_TF | X86_EFL_NT | X86_EFL_RF | X86_EFL_VM;
4806 switch (Idte.Gate.u4Type)
4807 {
4808 case AMD64_SEL_TYPE_SYS_INT_GATE:
4809 fEflToClear |= X86_EFL_IF;
4810 break;
4811 case AMD64_SEL_TYPE_SYS_TRAP_GATE:
4812 break;
4813
4814 default:
4815 Log(("iemRaiseXcptOrIntInLongMode %#x - invalid type (%#x) -> #GP\n", u8Vector, Idte.Gate.u4Type));
4816 return iemRaiseGeneralProtectionFault(pVCpu, X86_TRAP_ERR_IDT | ((uint16_t)u8Vector << X86_TRAP_ERR_SEL_SHIFT));
4817 }
4818
4819 /* Check DPL against CPL if applicable. */
4820 if (fFlags & IEM_XCPT_FLAGS_T_SOFT_INT)
4821 {
4822 if (pVCpu->iem.s.uCpl > Idte.Gate.u2Dpl)
4823 {
4824 Log(("iemRaiseXcptOrIntInLongMode %#x - CPL (%d) > DPL (%d) -> #GP\n", u8Vector, pVCpu->iem.s.uCpl, Idte.Gate.u2Dpl));
4825 return iemRaiseGeneralProtectionFault(pVCpu, X86_TRAP_ERR_IDT | ((uint16_t)u8Vector << X86_TRAP_ERR_SEL_SHIFT));
4826 }
4827 }
4828
4829 /* Is it there? */
4830 if (!Idte.Gate.u1Present)
4831 {
4832 Log(("iemRaiseXcptOrIntInLongMode %#x - not present -> #NP\n", u8Vector));
4833 return iemRaiseSelectorNotPresentWithErr(pVCpu, X86_TRAP_ERR_IDT | ((uint16_t)u8Vector << X86_TRAP_ERR_SEL_SHIFT));
4834 }
4835
4836 /* A null CS is bad. */
4837 RTSEL NewCS = Idte.Gate.u16Sel;
4838 if (!(NewCS & X86_SEL_MASK_OFF_RPL))
4839 {
4840 Log(("iemRaiseXcptOrIntInLongMode %#x - CS=%#x -> #GP\n", u8Vector, NewCS));
4841 return iemRaiseGeneralProtectionFault0(pVCpu);
4842 }
4843
4844 /* Fetch the descriptor for the new CS. */
4845 IEMSELDESC DescCS;
4846 rcStrict = iemMemFetchSelDesc(pVCpu, &DescCS, NewCS, X86_XCPT_GP);
4847 if (rcStrict != VINF_SUCCESS)
4848 {
4849 Log(("iemRaiseXcptOrIntInLongMode %#x - CS=%#x - rc=%Rrc\n", u8Vector, NewCS, VBOXSTRICTRC_VAL(rcStrict)));
4850 return rcStrict;
4851 }
4852
4853 /* Must be a 64-bit code segment. */
4854 if (!DescCS.Long.Gen.u1DescType)
4855 {
4856 Log(("iemRaiseXcptOrIntInLongMode %#x - CS=%#x - system selector (%#x) -> #GP\n", u8Vector, NewCS, DescCS.Legacy.Gen.u4Type));
4857 return iemRaiseGeneralProtectionFault(pVCpu, NewCS & X86_SEL_MASK_OFF_RPL);
4858 }
4859 if ( !DescCS.Long.Gen.u1Long
4860 || DescCS.Long.Gen.u1DefBig
4861 || !(DescCS.Long.Gen.u4Type & X86_SEL_TYPE_CODE) )
4862 {
4863 Log(("iemRaiseXcptOrIntInLongMode %#x - CS=%#x - not 64-bit code selector (%#x, L=%u, D=%u) -> #GP\n",
4864 u8Vector, NewCS, DescCS.Legacy.Gen.u4Type, DescCS.Long.Gen.u1Long, DescCS.Long.Gen.u1DefBig));
4865 return iemRaiseGeneralProtectionFault(pVCpu, NewCS & X86_SEL_MASK_OFF_RPL);
4866 }
4867
4868 /* Don't allow lowering the privilege level. For non-conforming CS
4869 selectors, the CS.DPL sets the privilege level the trap/interrupt
4870 handler runs at. For conforming CS selectors, the CPL remains
4871 unchanged, but the CS.DPL must be <= CPL. */
4872 /** @todo Testcase: Interrupt handler with CS.DPL=1, interrupt dispatched
4873 * when CPU in Ring-0. Result \#GP? */
4874 if (DescCS.Legacy.Gen.u2Dpl > pVCpu->iem.s.uCpl)
4875 {
4876 Log(("iemRaiseXcptOrIntInLongMode %#x - CS=%#x - DPL (%d) > CPL (%d) -> #GP\n",
4877 u8Vector, NewCS, DescCS.Legacy.Gen.u2Dpl, pVCpu->iem.s.uCpl));
4878 return iemRaiseGeneralProtectionFault(pVCpu, NewCS & X86_SEL_MASK_OFF_RPL);
4879 }
4880
4881
4882 /* Make sure the selector is present. */
4883 if (!DescCS.Legacy.Gen.u1Present)
4884 {
4885 Log(("iemRaiseXcptOrIntInLongMode %#x - CS=%#x - segment not present -> #NP\n", u8Vector, NewCS));
4886 return iemRaiseSelectorNotPresentBySelector(pVCpu, NewCS);
4887 }
4888
4889 /* Check that the new RIP is canonical. */
4890 uint64_t const uNewRip = Idte.Gate.u16OffsetLow
4891 | ((uint32_t)Idte.Gate.u16OffsetHigh << 16)
4892 | ((uint64_t)Idte.Gate.u32OffsetTop << 32);
4893 if (!IEM_IS_CANONICAL(uNewRip))
4894 {
4895 Log(("iemRaiseXcptOrIntInLongMode %#x - RIP=%#RX64 - Not canonical -> #GP(0)\n", u8Vector, uNewRip));
4896 return iemRaiseGeneralProtectionFault0(pVCpu);
4897 }
4898
4899 /*
4900 * If the privilege level changes or if the IST isn't zero, we need to get
4901 * a new stack from the TSS.
4902 */
4903 uint64_t uNewRsp;
4904 uint8_t const uNewCpl = DescCS.Legacy.Gen.u4Type & X86_SEL_TYPE_CONF
4905 ? pVCpu->iem.s.uCpl : DescCS.Legacy.Gen.u2Dpl;
4906 if ( uNewCpl != pVCpu->iem.s.uCpl
4907 || Idte.Gate.u3IST != 0)
4908 {
4909 rcStrict = iemRaiseLoadStackFromTss64(pVCpu, pCtx, uNewCpl, Idte.Gate.u3IST, &uNewRsp);
4910 if (rcStrict != VINF_SUCCESS)
4911 return rcStrict;
4912 }
4913 else
4914 uNewRsp = pCtx->rsp;
4915 uNewRsp &= ~(uint64_t)0xf;
4916
4917 /*
4918 * Calc the flag image to push.
4919 */
4920 uint32_t fEfl = IEMMISC_GET_EFL(pVCpu, pCtx);
4921 if (fFlags & (IEM_XCPT_FLAGS_DRx_INSTR_BP | IEM_XCPT_FLAGS_T_SOFT_INT))
4922 fEfl &= ~X86_EFL_RF;
4923 else if (!IEM_FULL_VERIFICATION_REM_ENABLED(pVCpu))
4924 fEfl |= X86_EFL_RF; /* Vagueness is all I've found on this so far... */ /** @todo Automatically pushing EFLAGS.RF. */
4925
4926 /*
4927 * Start making changes.
4928 */
4929 /* Set the new CPL so that stack accesses use it. */
4930 uint8_t const uOldCpl = pVCpu->iem.s.uCpl;
4931 pVCpu->iem.s.uCpl = uNewCpl;
4932
4933 /* Create the stack frame. */
4934 uint32_t cbStackFrame = sizeof(uint64_t) * (5 + !!(fFlags & IEM_XCPT_FLAGS_ERR));
4935 RTPTRUNION uStackFrame;
4936 rcStrict = iemMemMap(pVCpu, &uStackFrame.pv, cbStackFrame, UINT8_MAX,
4937 uNewRsp - cbStackFrame, IEM_ACCESS_STACK_W | IEM_ACCESS_WHAT_SYS); /* _SYS is a hack ... */
4938 if (rcStrict != VINF_SUCCESS)
4939 return rcStrict;
4940 void * const pvStackFrame = uStackFrame.pv;
4941
4942 if (fFlags & IEM_XCPT_FLAGS_ERR)
4943 *uStackFrame.pu64++ = uErr;
4944 uStackFrame.pu64[0] = fFlags & IEM_XCPT_FLAGS_T_SOFT_INT ? pCtx->rip + cbInstr : pCtx->rip;
4945 uStackFrame.pu64[1] = (pCtx->cs.Sel & ~X86_SEL_RPL) | uOldCpl; /* CPL paranoia */
4946 uStackFrame.pu64[2] = fEfl;
4947 uStackFrame.pu64[3] = pCtx->rsp;
4948 uStackFrame.pu64[4] = pCtx->ss.Sel;
4949 rcStrict = iemMemCommitAndUnmap(pVCpu, pvStackFrame, IEM_ACCESS_STACK_W | IEM_ACCESS_WHAT_SYS);
4950 if (rcStrict != VINF_SUCCESS)
4951 return rcStrict;
4952
4953 /* Mark the CS selectors 'accessed' (hope this is the correct time). */
4954 /** @todo testcase: excatly _when_ are the accessed bits set - before or
4955 * after pushing the stack frame? (Write protect the gdt + stack to
4956 * find out.) */
4957 if (!(DescCS.Legacy.Gen.u4Type & X86_SEL_TYPE_ACCESSED))
4958 {
4959 rcStrict = iemMemMarkSelDescAccessed(pVCpu, NewCS);
4960 if (rcStrict != VINF_SUCCESS)
4961 return rcStrict;
4962 DescCS.Legacy.Gen.u4Type |= X86_SEL_TYPE_ACCESSED;
4963 }
4964
4965 /*
4966 * Start comitting the register changes.
4967 */
4968 /** @todo research/testcase: Figure out what VT-x and AMD-V loads into the
4969 * hidden registers when interrupting 32-bit or 16-bit code! */
4970 if (uNewCpl != uOldCpl)
4971 {
4972 pCtx->ss.Sel = 0 | uNewCpl;
4973 pCtx->ss.ValidSel = 0 | uNewCpl;
4974 pCtx->ss.fFlags = CPUMSELREG_FLAGS_VALID;
4975 pCtx->ss.u32Limit = UINT32_MAX;
4976 pCtx->ss.u64Base = 0;
4977 pCtx->ss.Attr.u = (uNewCpl << X86DESCATTR_DPL_SHIFT) | X86DESCATTR_UNUSABLE;
4978 }
4979 pCtx->rsp = uNewRsp - cbStackFrame;
4980 pCtx->cs.Sel = (NewCS & ~X86_SEL_RPL) | uNewCpl;
4981 pCtx->cs.ValidSel = (NewCS & ~X86_SEL_RPL) | uNewCpl;
4982 pCtx->cs.fFlags = CPUMSELREG_FLAGS_VALID;
4983 pCtx->cs.u32Limit = X86DESC_LIMIT_G(&DescCS.Legacy);
4984 pCtx->cs.u64Base = X86DESC_BASE(&DescCS.Legacy);
4985 pCtx->cs.Attr.u = X86DESC_GET_HID_ATTR(&DescCS.Legacy);
4986 pCtx->rip = uNewRip;
4987
4988 fEfl &= ~fEflToClear;
4989 IEMMISC_SET_EFL(pVCpu, pCtx, fEfl);
4990
4991 if (fFlags & IEM_XCPT_FLAGS_CR2)
4992 pCtx->cr2 = uCr2;
4993
4994 if (fFlags & IEM_XCPT_FLAGS_T_CPU_XCPT)
4995 iemRaiseXcptAdjustState(pCtx, u8Vector);
4996
4997 return fFlags & IEM_XCPT_FLAGS_T_CPU_XCPT ? VINF_IEM_RAISED_XCPT : VINF_SUCCESS;
4998}
4999
5000
5001/**
5002 * Implements exceptions and interrupts.
5003 *
5004 * All exceptions and interrupts goes thru this function!
5005 *
5006 * @returns VBox strict status code.
5007 * @param pVCpu The cross context virtual CPU structure of the calling thread.
5008 * @param cbInstr The number of bytes to offset rIP by in the return
5009 * address.
5010 * @param u8Vector The interrupt / exception vector number.
5011 * @param fFlags The flags.
5012 * @param uErr The error value if IEM_XCPT_FLAGS_ERR is set.
5013 * @param uCr2 The CR2 value if IEM_XCPT_FLAGS_CR2 is set.
5014 */
5015DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC)
5016iemRaiseXcptOrInt(PVMCPU pVCpu,
5017 uint8_t cbInstr,
5018 uint8_t u8Vector,
5019 uint32_t fFlags,
5020 uint16_t uErr,
5021 uint64_t uCr2)
5022{
5023 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
5024#ifdef IN_RING0
5025 int rc = HMR0EnsureCompleteBasicContext(pVCpu, pCtx);
5026 AssertRCReturn(rc, rc);
5027#endif
5028
5029#ifndef IEM_WITH_CODE_TLB /** @todo we're doing it afterwards too, that should suffice... */
5030 /*
5031 * Flush prefetch buffer
5032 */
5033 pVCpu->iem.s.cbOpcode = pVCpu->iem.s.offOpcode;
5034#endif
5035
5036 /*
5037 * Perform the V8086 IOPL check and upgrade the fault without nesting.
5038 */
5039 if ( pCtx->eflags.Bits.u1VM
5040 && pCtx->eflags.Bits.u2IOPL != 3
5041 && (fFlags & (IEM_XCPT_FLAGS_T_SOFT_INT | IEM_XCPT_FLAGS_BP_INSTR)) == IEM_XCPT_FLAGS_T_SOFT_INT
5042 && (pCtx->cr0 & X86_CR0_PE) )
5043 {
5044 Log(("iemRaiseXcptOrInt: V8086 IOPL check failed for int %#x -> #GP(0)\n", u8Vector));
5045 fFlags = IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR;
5046 u8Vector = X86_XCPT_GP;
5047 uErr = 0;
5048 }
5049#ifdef DBGFTRACE_ENABLED
5050 RTTraceBufAddMsgF(pVCpu->CTX_SUFF(pVM)->CTX_SUFF(hTraceBuf), "Xcpt/%u: %02x %u %x %x %llx %04x:%04llx %04x:%04llx",
5051 pVCpu->iem.s.cXcptRecursions, u8Vector, cbInstr, fFlags, uErr, uCr2,
5052 pCtx->cs.Sel, pCtx->rip, pCtx->ss.Sel, pCtx->rsp);
5053#endif
5054
5055 /*
5056 * Do recursion accounting.
5057 */
5058 uint8_t const uPrevXcpt = pVCpu->iem.s.uCurXcpt;
5059 uint32_t const fPrevXcpt = pVCpu->iem.s.fCurXcpt;
5060 if (pVCpu->iem.s.cXcptRecursions == 0)
5061 Log(("iemRaiseXcptOrInt: %#x at %04x:%RGv cbInstr=%#x fFlags=%#x uErr=%#x uCr2=%llx\n",
5062 u8Vector, pCtx->cs.Sel, pCtx->rip, cbInstr, fFlags, uErr, uCr2));
5063 else
5064 {
5065 Log(("iemRaiseXcptOrInt: %#x at %04x:%RGv cbInstr=%#x fFlags=%#x uErr=%#x uCr2=%llx; prev=%#x depth=%d flags=%#x\n",
5066 u8Vector, pCtx->cs.Sel, pCtx->rip, cbInstr, fFlags, uErr, uCr2, pVCpu->iem.s.uCurXcpt, pVCpu->iem.s.cXcptRecursions + 1, fPrevXcpt));
5067
5068 /** @todo double and tripple faults. */
5069 if (pVCpu->iem.s.cXcptRecursions >= 3)
5070 {
5071#ifdef DEBUG_bird
5072 AssertFailed();
5073#endif
5074 IEM_RETURN_ASPECT_NOT_IMPLEMENTED_LOG(("Too many fault nestings.\n"));
5075 }
5076
5077 /** @todo set X86_TRAP_ERR_EXTERNAL when appropriate.
5078 if (fPrevXcpt & IEM_XCPT_FLAGS_T_EXT_INT)
5079 {
5080 ....
5081 } */
5082 }
5083 pVCpu->iem.s.cXcptRecursions++;
5084 pVCpu->iem.s.uCurXcpt = u8Vector;
5085 pVCpu->iem.s.fCurXcpt = fFlags;
5086
5087 /*
5088 * Extensive logging.
5089 */
5090#if defined(LOG_ENABLED) && defined(IN_RING3)
5091 if (LogIs3Enabled())
5092 {
5093 PVM pVM = pVCpu->CTX_SUFF(pVM);
5094 char szRegs[4096];
5095 DBGFR3RegPrintf(pVM->pUVM, pVCpu->idCpu, &szRegs[0], sizeof(szRegs),
5096 "rax=%016VR{rax} rbx=%016VR{rbx} rcx=%016VR{rcx} rdx=%016VR{rdx}\n"
5097 "rsi=%016VR{rsi} rdi=%016VR{rdi} r8 =%016VR{r8} r9 =%016VR{r9}\n"
5098 "r10=%016VR{r10} r11=%016VR{r11} r12=%016VR{r12} r13=%016VR{r13}\n"
5099 "r14=%016VR{r14} r15=%016VR{r15} %VRF{rflags}\n"
5100 "rip=%016VR{rip} rsp=%016VR{rsp} rbp=%016VR{rbp}\n"
5101 "cs={%04VR{cs} base=%016VR{cs_base} limit=%08VR{cs_lim} flags=%04VR{cs_attr}} cr0=%016VR{cr0}\n"
5102 "ds={%04VR{ds} base=%016VR{ds_base} limit=%08VR{ds_lim} flags=%04VR{ds_attr}} cr2=%016VR{cr2}\n"
5103 "es={%04VR{es} base=%016VR{es_base} limit=%08VR{es_lim} flags=%04VR{es_attr}} cr3=%016VR{cr3}\n"
5104 "fs={%04VR{fs} base=%016VR{fs_base} limit=%08VR{fs_lim} flags=%04VR{fs_attr}} cr4=%016VR{cr4}\n"
5105 "gs={%04VR{gs} base=%016VR{gs_base} limit=%08VR{gs_lim} flags=%04VR{gs_attr}} cr8=%016VR{cr8}\n"
5106 "ss={%04VR{ss} base=%016VR{ss_base} limit=%08VR{ss_lim} flags=%04VR{ss_attr}}\n"
5107 "dr0=%016VR{dr0} dr1=%016VR{dr1} dr2=%016VR{dr2} dr3=%016VR{dr3}\n"
5108 "dr6=%016VR{dr6} dr7=%016VR{dr7}\n"
5109 "gdtr=%016VR{gdtr_base}:%04VR{gdtr_lim} idtr=%016VR{idtr_base}:%04VR{idtr_lim} rflags=%08VR{rflags}\n"
5110 "ldtr={%04VR{ldtr} base=%016VR{ldtr_base} limit=%08VR{ldtr_lim} flags=%08VR{ldtr_attr}}\n"
5111 "tr ={%04VR{tr} base=%016VR{tr_base} limit=%08VR{tr_lim} flags=%08VR{tr_attr}}\n"
5112 " sysenter={cs=%04VR{sysenter_cs} eip=%08VR{sysenter_eip} esp=%08VR{sysenter_esp}}\n"
5113 " efer=%016VR{efer}\n"
5114 " pat=%016VR{pat}\n"
5115 " sf_mask=%016VR{sf_mask}\n"
5116 "krnl_gs_base=%016VR{krnl_gs_base}\n"
5117 " lstar=%016VR{lstar}\n"
5118 " star=%016VR{star} cstar=%016VR{cstar}\n"
5119 "fcw=%04VR{fcw} fsw=%04VR{fsw} ftw=%04VR{ftw} mxcsr=%04VR{mxcsr} mxcsr_mask=%04VR{mxcsr_mask}\n"
5120 );
5121
5122 char szInstr[256];
5123 DBGFR3DisasInstrEx(pVM->pUVM, pVCpu->idCpu, 0, 0,
5124 DBGF_DISAS_FLAGS_CURRENT_GUEST | DBGF_DISAS_FLAGS_DEFAULT_MODE,
5125 szInstr, sizeof(szInstr), NULL);
5126 Log3(("%s%s\n", szRegs, szInstr));
5127 }
5128#endif /* LOG_ENABLED */
5129
5130 /*
5131 * Call the mode specific worker function.
5132 */
5133 VBOXSTRICTRC rcStrict;
5134 if (!(pCtx->cr0 & X86_CR0_PE))
5135 rcStrict = iemRaiseXcptOrIntInRealMode(pVCpu, pCtx, cbInstr, u8Vector, fFlags, uErr, uCr2);
5136 else if (pCtx->msrEFER & MSR_K6_EFER_LMA)
5137 rcStrict = iemRaiseXcptOrIntInLongMode(pVCpu, pCtx, cbInstr, u8Vector, fFlags, uErr, uCr2);
5138 else
5139 rcStrict = iemRaiseXcptOrIntInProtMode(pVCpu, pCtx, cbInstr, u8Vector, fFlags, uErr, uCr2);
5140
5141 /* Flush the prefetch buffer. */
5142#ifdef IEM_WITH_CODE_TLB
5143 pVCpu->iem.s.pbInstrBuf = NULL;
5144#else
5145 pVCpu->iem.s.cbOpcode = IEM_GET_INSTR_LEN(pVCpu);
5146#endif
5147
5148 /*
5149 * Unwind.
5150 */
5151 pVCpu->iem.s.cXcptRecursions--;
5152 pVCpu->iem.s.uCurXcpt = uPrevXcpt;
5153 pVCpu->iem.s.fCurXcpt = fPrevXcpt;
5154 Log(("iemRaiseXcptOrInt: returns %Rrc (vec=%#x); cs:rip=%04x:%RGv ss:rsp=%04x:%RGv cpl=%u\n",
5155 VBOXSTRICTRC_VAL(rcStrict), u8Vector, pCtx->cs.Sel, pCtx->rip, pCtx->ss.Sel, pCtx->esp, pVCpu->iem.s.uCpl));
5156 return rcStrict;
5157}
5158
5159#ifdef IEM_WITH_SETJMP
5160/**
5161 * See iemRaiseXcptOrInt. Will not return.
5162 */
5163IEM_STATIC DECL_NO_RETURN(void)
5164iemRaiseXcptOrIntJmp(PVMCPU pVCpu,
5165 uint8_t cbInstr,
5166 uint8_t u8Vector,
5167 uint32_t fFlags,
5168 uint16_t uErr,
5169 uint64_t uCr2)
5170{
5171 VBOXSTRICTRC rcStrict = iemRaiseXcptOrInt(pVCpu, cbInstr, u8Vector, fFlags, uErr, uCr2);
5172 longjmp(*pVCpu->iem.s.CTX_SUFF(pJmpBuf), VBOXSTRICTRC_VAL(rcStrict));
5173}
5174#endif
5175
5176
5177/** \#DE - 00. */
5178DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseDivideError(PVMCPU pVCpu)
5179{
5180 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_DE, IEM_XCPT_FLAGS_T_CPU_XCPT, 0, 0);
5181}
5182
5183
5184/** \#DB - 01.
5185 * @note This automatically clear DR7.GD. */
5186DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseDebugException(PVMCPU pVCpu)
5187{
5188 /** @todo set/clear RF. */
5189 IEM_GET_CTX(pVCpu)->dr[7] &= ~X86_DR7_GD;
5190 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_DB, IEM_XCPT_FLAGS_T_CPU_XCPT, 0, 0);
5191}
5192
5193
5194/** \#UD - 06. */
5195DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseUndefinedOpcode(PVMCPU pVCpu)
5196{
5197 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_UD, IEM_XCPT_FLAGS_T_CPU_XCPT, 0, 0);
5198}
5199
5200
5201/** \#NM - 07. */
5202DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseDeviceNotAvailable(PVMCPU pVCpu)
5203{
5204 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_NM, IEM_XCPT_FLAGS_T_CPU_XCPT, 0, 0);
5205}
5206
5207
5208/** \#TS(err) - 0a. */
5209DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseTaskSwitchFaultWithErr(PVMCPU pVCpu, uint16_t uErr)
5210{
5211 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_TS, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR, uErr, 0);
5212}
5213
5214
5215/** \#TS(tr) - 0a. */
5216DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseTaskSwitchFaultCurrentTSS(PVMCPU pVCpu)
5217{
5218 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_TS, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR,
5219 IEM_GET_CTX(pVCpu)->tr.Sel, 0);
5220}
5221
5222
5223/** \#TS(0) - 0a. */
5224DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseTaskSwitchFault0(PVMCPU pVCpu)
5225{
5226 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_TS, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR,
5227 0, 0);
5228}
5229
5230
5231/** \#TS(err) - 0a. */
5232DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseTaskSwitchFaultBySelector(PVMCPU pVCpu, uint16_t uSel)
5233{
5234 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_TS, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR,
5235 uSel & X86_SEL_MASK_OFF_RPL, 0);
5236}
5237
5238
5239/** \#NP(err) - 0b. */
5240DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseSelectorNotPresentWithErr(PVMCPU pVCpu, uint16_t uErr)
5241{
5242 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_NP, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR, uErr, 0);
5243}
5244
5245
5246/** \#NP(seg) - 0b. */
5247DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseSelectorNotPresentBySegReg(PVMCPU pVCpu, uint32_t iSegReg)
5248{
5249 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_NP, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR,
5250 iemSRegFetchU16(pVCpu, iSegReg) & ~X86_SEL_RPL, 0);
5251}
5252
5253
5254/** \#NP(sel) - 0b. */
5255DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseSelectorNotPresentBySelector(PVMCPU pVCpu, uint16_t uSel)
5256{
5257 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_NP, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR,
5258 uSel & ~X86_SEL_RPL, 0);
5259}
5260
5261
5262/** \#SS(seg) - 0c. */
5263DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseStackSelectorNotPresentBySelector(PVMCPU pVCpu, uint16_t uSel)
5264{
5265 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_SS, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR,
5266 uSel & ~X86_SEL_RPL, 0);
5267}
5268
5269
5270/** \#SS(err) - 0c. */
5271DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseStackSelectorNotPresentWithErr(PVMCPU pVCpu, uint16_t uErr)
5272{
5273 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_SS, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR, uErr, 0);
5274}
5275
5276
5277/** \#GP(n) - 0d. */
5278DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseGeneralProtectionFault(PVMCPU pVCpu, uint16_t uErr)
5279{
5280 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_GP, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR, uErr, 0);
5281}
5282
5283
5284/** \#GP(0) - 0d. */
5285DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseGeneralProtectionFault0(PVMCPU pVCpu)
5286{
5287 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_GP, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR, 0, 0);
5288}
5289
5290#ifdef IEM_WITH_SETJMP
5291/** \#GP(0) - 0d. */
5292DECL_NO_INLINE(IEM_STATIC, DECL_NO_RETURN(void)) iemRaiseGeneralProtectionFault0Jmp(PVMCPU pVCpu)
5293{
5294 iemRaiseXcptOrIntJmp(pVCpu, 0, X86_XCPT_GP, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR, 0, 0);
5295}
5296#endif
5297
5298
5299/** \#GP(sel) - 0d. */
5300DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseGeneralProtectionFaultBySelector(PVMCPU pVCpu, RTSEL Sel)
5301{
5302 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_GP, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR,
5303 Sel & ~X86_SEL_RPL, 0);
5304}
5305
5306
5307/** \#GP(0) - 0d. */
5308DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseNotCanonical(PVMCPU pVCpu)
5309{
5310 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_GP, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR, 0, 0);
5311}
5312
5313
5314/** \#GP(sel) - 0d. */
5315DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseSelectorBounds(PVMCPU pVCpu, uint32_t iSegReg, uint32_t fAccess)
5316{
5317 NOREF(iSegReg); NOREF(fAccess);
5318 return iemRaiseXcptOrInt(pVCpu, 0, iSegReg == X86_SREG_SS ? X86_XCPT_SS : X86_XCPT_GP,
5319 IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR, 0, 0);
5320}
5321
5322#ifdef IEM_WITH_SETJMP
5323/** \#GP(sel) - 0d, longjmp. */
5324DECL_NO_INLINE(IEM_STATIC, DECL_NO_RETURN(void)) iemRaiseSelectorBoundsJmp(PVMCPU pVCpu, uint32_t iSegReg, uint32_t fAccess)
5325{
5326 NOREF(iSegReg); NOREF(fAccess);
5327 iemRaiseXcptOrIntJmp(pVCpu, 0, iSegReg == X86_SREG_SS ? X86_XCPT_SS : X86_XCPT_GP,
5328 IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR, 0, 0);
5329}
5330#endif
5331
5332/** \#GP(sel) - 0d. */
5333DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseSelectorBoundsBySelector(PVMCPU pVCpu, RTSEL Sel)
5334{
5335 NOREF(Sel);
5336 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_GP, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR, 0, 0);
5337}
5338
5339#ifdef IEM_WITH_SETJMP
5340/** \#GP(sel) - 0d, longjmp. */
5341DECL_NO_INLINE(IEM_STATIC, DECL_NO_RETURN(void)) iemRaiseSelectorBoundsBySelectorJmp(PVMCPU pVCpu, RTSEL Sel)
5342{
5343 NOREF(Sel);
5344 iemRaiseXcptOrIntJmp(pVCpu, 0, X86_XCPT_GP, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR, 0, 0);
5345}
5346#endif
5347
5348
5349/** \#GP(sel) - 0d. */
5350DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseSelectorInvalidAccess(PVMCPU pVCpu, uint32_t iSegReg, uint32_t fAccess)
5351{
5352 NOREF(iSegReg); NOREF(fAccess);
5353 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_GP, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR, 0, 0);
5354}
5355
5356#ifdef IEM_WITH_SETJMP
5357/** \#GP(sel) - 0d, longjmp. */
5358DECL_NO_INLINE(IEM_STATIC, DECL_NO_RETURN(void)) iemRaiseSelectorInvalidAccessJmp(PVMCPU pVCpu, uint32_t iSegReg,
5359 uint32_t fAccess)
5360{
5361 NOREF(iSegReg); NOREF(fAccess);
5362 iemRaiseXcptOrIntJmp(pVCpu, 0, X86_XCPT_GP, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR, 0, 0);
5363}
5364#endif
5365
5366
5367/** \#PF(n) - 0e. */
5368DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaisePageFault(PVMCPU pVCpu, RTGCPTR GCPtrWhere, uint32_t fAccess, int rc)
5369{
5370 uint16_t uErr;
5371 switch (rc)
5372 {
5373 case VERR_PAGE_NOT_PRESENT:
5374 case VERR_PAGE_TABLE_NOT_PRESENT:
5375 case VERR_PAGE_DIRECTORY_PTR_NOT_PRESENT:
5376 case VERR_PAGE_MAP_LEVEL4_NOT_PRESENT:
5377 uErr = 0;
5378 break;
5379
5380 default:
5381 AssertMsgFailed(("%Rrc\n", rc));
5382 case VERR_ACCESS_DENIED:
5383 uErr = X86_TRAP_PF_P;
5384 break;
5385
5386 /** @todo reserved */
5387 }
5388
5389 if (pVCpu->iem.s.uCpl == 3)
5390 uErr |= X86_TRAP_PF_US;
5391
5392 if ( (fAccess & IEM_ACCESS_WHAT_MASK) == IEM_ACCESS_WHAT_CODE
5393 && ( (IEM_GET_CTX(pVCpu)->cr4 & X86_CR4_PAE)
5394 && (IEM_GET_CTX(pVCpu)->msrEFER & MSR_K6_EFER_NXE) ) )
5395 uErr |= X86_TRAP_PF_ID;
5396
5397#if 0 /* This is so much non-sense, really. Why was it done like that? */
5398 /* Note! RW access callers reporting a WRITE protection fault, will clear
5399 the READ flag before calling. So, read-modify-write accesses (RW)
5400 can safely be reported as READ faults. */
5401 if ((fAccess & (IEM_ACCESS_TYPE_WRITE | IEM_ACCESS_TYPE_READ)) == IEM_ACCESS_TYPE_WRITE)
5402 uErr |= X86_TRAP_PF_RW;
5403#else
5404 if (fAccess & IEM_ACCESS_TYPE_WRITE)
5405 {
5406 if (!IEM_FULL_VERIFICATION_REM_ENABLED(pVCpu) || !(fAccess & IEM_ACCESS_TYPE_READ))
5407 uErr |= X86_TRAP_PF_RW;
5408 }
5409#endif
5410
5411 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_PF, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR | IEM_XCPT_FLAGS_CR2,
5412 uErr, GCPtrWhere);
5413}
5414
5415#ifdef IEM_WITH_SETJMP
5416/** \#PF(n) - 0e, longjmp. */
5417IEM_STATIC DECL_NO_RETURN(void) iemRaisePageFaultJmp(PVMCPU pVCpu, RTGCPTR GCPtrWhere, uint32_t fAccess, int rc)
5418{
5419 longjmp(*CTX_SUFF(pVCpu->iem.s.pJmpBuf), VBOXSTRICTRC_VAL(iemRaisePageFault(pVCpu, GCPtrWhere, fAccess, rc)));
5420}
5421#endif
5422
5423
5424/** \#MF(0) - 10. */
5425DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseMathFault(PVMCPU pVCpu)
5426{
5427 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_MF, IEM_XCPT_FLAGS_T_CPU_XCPT, 0, 0);
5428}
5429
5430
5431/** \#AC(0) - 11. */
5432DECL_NO_INLINE(IEM_STATIC, VBOXSTRICTRC) iemRaiseAlignmentCheckException(PVMCPU pVCpu)
5433{
5434 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_AC, IEM_XCPT_FLAGS_T_CPU_XCPT, 0, 0);
5435}
5436
5437
5438/**
5439 * Macro for calling iemCImplRaiseDivideError().
5440 *
5441 * This enables us to add/remove arguments and force different levels of
5442 * inlining as we wish.
5443 *
5444 * @return Strict VBox status code.
5445 */
5446#define IEMOP_RAISE_DIVIDE_ERROR() IEM_MC_DEFER_TO_CIMPL_0(iemCImplRaiseDivideError)
5447IEM_CIMPL_DEF_0(iemCImplRaiseDivideError)
5448{
5449 NOREF(cbInstr);
5450 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_DE, IEM_XCPT_FLAGS_T_CPU_XCPT, 0, 0);
5451}
5452
5453
5454/**
5455 * Macro for calling iemCImplRaiseInvalidLockPrefix().
5456 *
5457 * This enables us to add/remove arguments and force different levels of
5458 * inlining as we wish.
5459 *
5460 * @return Strict VBox status code.
5461 */
5462#define IEMOP_RAISE_INVALID_LOCK_PREFIX() IEM_MC_DEFER_TO_CIMPL_0(iemCImplRaiseInvalidLockPrefix)
5463IEM_CIMPL_DEF_0(iemCImplRaiseInvalidLockPrefix)
5464{
5465 NOREF(cbInstr);
5466 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_UD, IEM_XCPT_FLAGS_T_CPU_XCPT, 0, 0);
5467}
5468
5469
5470/**
5471 * Macro for calling iemCImplRaiseInvalidOpcode().
5472 *
5473 * This enables us to add/remove arguments and force different levels of
5474 * inlining as we wish.
5475 *
5476 * @return Strict VBox status code.
5477 */
5478#define IEMOP_RAISE_INVALID_OPCODE() IEM_MC_DEFER_TO_CIMPL_0(iemCImplRaiseInvalidOpcode)
5479IEM_CIMPL_DEF_0(iemCImplRaiseInvalidOpcode)
5480{
5481 NOREF(cbInstr);
5482 return iemRaiseXcptOrInt(pVCpu, 0, X86_XCPT_UD, IEM_XCPT_FLAGS_T_CPU_XCPT, 0, 0);
5483}
5484
5485
5486/** @} */
5487
5488
5489/*
5490 *
5491 * Helpers routines.
5492 * Helpers routines.
5493 * Helpers routines.
5494 *
5495 */
5496
5497/**
5498 * Recalculates the effective operand size.
5499 *
5500 * @param pVCpu The cross context virtual CPU structure of the calling thread.
5501 */
5502IEM_STATIC void iemRecalEffOpSize(PVMCPU pVCpu)
5503{
5504 switch (pVCpu->iem.s.enmCpuMode)
5505 {
5506 case IEMMODE_16BIT:
5507 pVCpu->iem.s.enmEffOpSize = pVCpu->iem.s.fPrefixes & IEM_OP_PRF_SIZE_OP ? IEMMODE_32BIT : IEMMODE_16BIT;
5508 break;
5509 case IEMMODE_32BIT:
5510 pVCpu->iem.s.enmEffOpSize = pVCpu->iem.s.fPrefixes & IEM_OP_PRF_SIZE_OP ? IEMMODE_16BIT : IEMMODE_32BIT;
5511 break;
5512 case IEMMODE_64BIT:
5513 switch (pVCpu->iem.s.fPrefixes & (IEM_OP_PRF_SIZE_REX_W | IEM_OP_PRF_SIZE_OP))
5514 {
5515 case 0:
5516 pVCpu->iem.s.enmEffOpSize = pVCpu->iem.s.enmDefOpSize;
5517 break;
5518 case IEM_OP_PRF_SIZE_OP:
5519 pVCpu->iem.s.enmEffOpSize = IEMMODE_16BIT;
5520 break;
5521 case IEM_OP_PRF_SIZE_REX_W:
5522 case IEM_OP_PRF_SIZE_REX_W | IEM_OP_PRF_SIZE_OP:
5523 pVCpu->iem.s.enmEffOpSize = IEMMODE_64BIT;
5524 break;
5525 }
5526 break;
5527 default:
5528 AssertFailed();
5529 }
5530}
5531
5532
5533/**
5534 * Sets the default operand size to 64-bit and recalculates the effective
5535 * operand size.
5536 *
5537 * @param pVCpu The cross context virtual CPU structure of the calling thread.
5538 */
5539IEM_STATIC void iemRecalEffOpSize64Default(PVMCPU pVCpu)
5540{
5541 Assert(pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT);
5542 pVCpu->iem.s.enmDefOpSize = IEMMODE_64BIT;
5543 if ((pVCpu->iem.s.fPrefixes & (IEM_OP_PRF_SIZE_REX_W | IEM_OP_PRF_SIZE_OP)) != IEM_OP_PRF_SIZE_OP)
5544 pVCpu->iem.s.enmEffOpSize = IEMMODE_64BIT;
5545 else
5546 pVCpu->iem.s.enmEffOpSize = IEMMODE_16BIT;
5547}
5548
5549
5550/*
5551 *
5552 * Common opcode decoders.
5553 * Common opcode decoders.
5554 * Common opcode decoders.
5555 *
5556 */
5557//#include <iprt/mem.h>
5558
5559/**
5560 * Used to add extra details about a stub case.
5561 * @param pVCpu The cross context virtual CPU structure of the calling thread.
5562 */
5563IEM_STATIC void iemOpStubMsg2(PVMCPU pVCpu)
5564{
5565#if defined(LOG_ENABLED) && defined(IN_RING3)
5566 PVM pVM = pVCpu->CTX_SUFF(pVM);
5567 char szRegs[4096];
5568 DBGFR3RegPrintf(pVM->pUVM, pVCpu->idCpu, &szRegs[0], sizeof(szRegs),
5569 "rax=%016VR{rax} rbx=%016VR{rbx} rcx=%016VR{rcx} rdx=%016VR{rdx}\n"
5570 "rsi=%016VR{rsi} rdi=%016VR{rdi} r8 =%016VR{r8} r9 =%016VR{r9}\n"
5571 "r10=%016VR{r10} r11=%016VR{r11} r12=%016VR{r12} r13=%016VR{r13}\n"
5572 "r14=%016VR{r14} r15=%016VR{r15} %VRF{rflags}\n"
5573 "rip=%016VR{rip} rsp=%016VR{rsp} rbp=%016VR{rbp}\n"
5574 "cs={%04VR{cs} base=%016VR{cs_base} limit=%08VR{cs_lim} flags=%04VR{cs_attr}} cr0=%016VR{cr0}\n"
5575 "ds={%04VR{ds} base=%016VR{ds_base} limit=%08VR{ds_lim} flags=%04VR{ds_attr}} cr2=%016VR{cr2}\n"
5576 "es={%04VR{es} base=%016VR{es_base} limit=%08VR{es_lim} flags=%04VR{es_attr}} cr3=%016VR{cr3}\n"
5577 "fs={%04VR{fs} base=%016VR{fs_base} limit=%08VR{fs_lim} flags=%04VR{fs_attr}} cr4=%016VR{cr4}\n"
5578 "gs={%04VR{gs} base=%016VR{gs_base} limit=%08VR{gs_lim} flags=%04VR{gs_attr}} cr8=%016VR{cr8}\n"
5579 "ss={%04VR{ss} base=%016VR{ss_base} limit=%08VR{ss_lim} flags=%04VR{ss_attr}}\n"
5580 "dr0=%016VR{dr0} dr1=%016VR{dr1} dr2=%016VR{dr2} dr3=%016VR{dr3}\n"
5581 "dr6=%016VR{dr6} dr7=%016VR{dr7}\n"
5582 "gdtr=%016VR{gdtr_base}:%04VR{gdtr_lim} idtr=%016VR{idtr_base}:%04VR{idtr_lim} rflags=%08VR{rflags}\n"
5583 "ldtr={%04VR{ldtr} base=%016VR{ldtr_base} limit=%08VR{ldtr_lim} flags=%08VR{ldtr_attr}}\n"
5584 "tr ={%04VR{tr} base=%016VR{tr_base} limit=%08VR{tr_lim} flags=%08VR{tr_attr}}\n"
5585 " sysenter={cs=%04VR{sysenter_cs} eip=%08VR{sysenter_eip} esp=%08VR{sysenter_esp}}\n"
5586 " efer=%016VR{efer}\n"
5587 " pat=%016VR{pat}\n"
5588 " sf_mask=%016VR{sf_mask}\n"
5589 "krnl_gs_base=%016VR{krnl_gs_base}\n"
5590 " lstar=%016VR{lstar}\n"
5591 " star=%016VR{star} cstar=%016VR{cstar}\n"
5592 "fcw=%04VR{fcw} fsw=%04VR{fsw} ftw=%04VR{ftw} mxcsr=%04VR{mxcsr} mxcsr_mask=%04VR{mxcsr_mask}\n"
5593 );
5594
5595 char szInstr[256];
5596 DBGFR3DisasInstrEx(pVM->pUVM, pVCpu->idCpu, 0, 0,
5597 DBGF_DISAS_FLAGS_CURRENT_GUEST | DBGF_DISAS_FLAGS_DEFAULT_MODE,
5598 szInstr, sizeof(szInstr), NULL);
5599
5600 RTAssertMsg2Weak("%s%s\n", szRegs, szInstr);
5601#else
5602 RTAssertMsg2Weak("cs:rip=%04x:%RX64\n", IEM_GET_CTX(pVCpu)->cs, IEM_GET_CTX(pVCpu)->rip);
5603#endif
5604}
5605
5606/**
5607 * Complains about a stub.
5608 *
5609 * Providing two versions of this macro, one for daily use and one for use when
5610 * working on IEM.
5611 */
5612#if 0
5613# define IEMOP_BITCH_ABOUT_STUB() \
5614 do { \
5615 RTAssertMsg1(NULL, __LINE__, __FILE__, __FUNCTION__); \
5616 iemOpStubMsg2(pVCpu); \
5617 RTAssertPanic(); \
5618 } while (0)
5619#else
5620# define IEMOP_BITCH_ABOUT_STUB() Log(("Stub: %s (line %d)\n", __FUNCTION__, __LINE__));
5621#endif
5622
5623/** Stubs an opcode. */
5624#define FNIEMOP_STUB(a_Name) \
5625 FNIEMOP_DEF(a_Name) \
5626 { \
5627 RT_NOREF_PV(pVCpu); \
5628 IEMOP_BITCH_ABOUT_STUB(); \
5629 return VERR_IEM_INSTR_NOT_IMPLEMENTED; \
5630 } \
5631 typedef int ignore_semicolon
5632
5633/** Stubs an opcode. */
5634#define FNIEMOP_STUB_1(a_Name, a_Type0, a_Name0) \
5635 FNIEMOP_DEF_1(a_Name, a_Type0, a_Name0) \
5636 { \
5637 RT_NOREF_PV(pVCpu); \
5638 RT_NOREF_PV(a_Name0); \
5639 IEMOP_BITCH_ABOUT_STUB(); \
5640 return VERR_IEM_INSTR_NOT_IMPLEMENTED; \
5641 } \
5642 typedef int ignore_semicolon
5643
5644/** Stubs an opcode which currently should raise \#UD. */
5645#define FNIEMOP_UD_STUB(a_Name) \
5646 FNIEMOP_DEF(a_Name) \
5647 { \
5648 Log(("Unsupported instruction %Rfn\n", __FUNCTION__)); \
5649 return IEMOP_RAISE_INVALID_OPCODE(); \
5650 } \
5651 typedef int ignore_semicolon
5652
5653/** Stubs an opcode which currently should raise \#UD. */
5654#define FNIEMOP_UD_STUB_1(a_Name, a_Type0, a_Name0) \
5655 FNIEMOP_DEF_1(a_Name, a_Type0, a_Name0) \
5656 { \
5657 RT_NOREF_PV(pVCpu); \
5658 RT_NOREF_PV(a_Name0); \
5659 Log(("Unsupported instruction %Rfn\n", __FUNCTION__)); \
5660 return IEMOP_RAISE_INVALID_OPCODE(); \
5661 } \
5662 typedef int ignore_semicolon
5663
5664
5665
5666/** @name Register Access.
5667 * @{
5668 */
5669
5670/**
5671 * Gets a reference (pointer) to the specified hidden segment register.
5672 *
5673 * @returns Hidden register reference.
5674 * @param pVCpu The cross context virtual CPU structure of the calling thread.
5675 * @param iSegReg The segment register.
5676 */
5677IEM_STATIC PCPUMSELREG iemSRegGetHid(PVMCPU pVCpu, uint8_t iSegReg)
5678{
5679 Assert(iSegReg < X86_SREG_COUNT);
5680 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
5681 PCPUMSELREG pSReg = &pCtx->aSRegs[iSegReg];
5682
5683#ifdef VBOX_WITH_RAW_MODE_NOT_R0
5684 if (RT_LIKELY(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, pSReg)))
5685 { /* likely */ }
5686 else
5687 CPUMGuestLazyLoadHiddenSelectorReg(pVCpu, pSReg);
5688#else
5689 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, pSReg));
5690#endif
5691 return pSReg;
5692}
5693
5694
5695/**
5696 * Ensures that the given hidden segment register is up to date.
5697 *
5698 * @returns Hidden register reference.
5699 * @param pVCpu The cross context virtual CPU structure of the calling thread.
5700 * @param pSReg The segment register.
5701 */
5702IEM_STATIC PCPUMSELREG iemSRegUpdateHid(PVMCPU pVCpu, PCPUMSELREG pSReg)
5703{
5704#ifdef VBOX_WITH_RAW_MODE_NOT_R0
5705 if (!CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, pSReg))
5706 CPUMGuestLazyLoadHiddenSelectorReg(pVCpu, pSReg);
5707#else
5708 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, pSReg));
5709 NOREF(pVCpu);
5710#endif
5711 return pSReg;
5712}
5713
5714
5715/**
5716 * Gets a reference (pointer) to the specified segment register (the selector
5717 * value).
5718 *
5719 * @returns Pointer to the selector variable.
5720 * @param pVCpu The cross context virtual CPU structure of the calling thread.
5721 * @param iSegReg The segment register.
5722 */
5723DECLINLINE(uint16_t *) iemSRegRef(PVMCPU pVCpu, uint8_t iSegReg)
5724{
5725 Assert(iSegReg < X86_SREG_COUNT);
5726 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
5727 return &pCtx->aSRegs[iSegReg].Sel;
5728}
5729
5730
5731/**
5732 * Fetches the selector value of a segment register.
5733 *
5734 * @returns The selector value.
5735 * @param pVCpu The cross context virtual CPU structure of the calling thread.
5736 * @param iSegReg The segment register.
5737 */
5738DECLINLINE(uint16_t) iemSRegFetchU16(PVMCPU pVCpu, uint8_t iSegReg)
5739{
5740 Assert(iSegReg < X86_SREG_COUNT);
5741 return IEM_GET_CTX(pVCpu)->aSRegs[iSegReg].Sel;
5742}
5743
5744
5745/**
5746 * Gets a reference (pointer) to the specified general purpose register.
5747 *
5748 * @returns Register reference.
5749 * @param pVCpu The cross context virtual CPU structure of the calling thread.
5750 * @param iReg The general purpose register.
5751 */
5752DECLINLINE(void *) iemGRegRef(PVMCPU pVCpu, uint8_t iReg)
5753{
5754 Assert(iReg < 16);
5755 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
5756 return &pCtx->aGRegs[iReg];
5757}
5758
5759
5760/**
5761 * Gets a reference (pointer) to the specified 8-bit general purpose register.
5762 *
5763 * Because of AH, CH, DH and BH we cannot use iemGRegRef directly here.
5764 *
5765 * @returns Register reference.
5766 * @param pVCpu The cross context virtual CPU structure of the calling thread.
5767 * @param iReg The register.
5768 */
5769DECLINLINE(uint8_t *) iemGRegRefU8(PVMCPU pVCpu, uint8_t iReg)
5770{
5771 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
5772 if (iReg < 4 || (pVCpu->iem.s.fPrefixes & IEM_OP_PRF_REX))
5773 {
5774 Assert(iReg < 16);
5775 return &pCtx->aGRegs[iReg].u8;
5776 }
5777 /* high 8-bit register. */
5778 Assert(iReg < 8);
5779 return &pCtx->aGRegs[iReg & 3].bHi;
5780}
5781
5782
5783/**
5784 * Gets a reference (pointer) to the specified 16-bit general purpose register.
5785 *
5786 * @returns Register reference.
5787 * @param pVCpu The cross context virtual CPU structure of the calling thread.
5788 * @param iReg The register.
5789 */
5790DECLINLINE(uint16_t *) iemGRegRefU16(PVMCPU pVCpu, uint8_t iReg)
5791{
5792 Assert(iReg < 16);
5793 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
5794 return &pCtx->aGRegs[iReg].u16;
5795}
5796
5797
5798/**
5799 * Gets a reference (pointer) to the specified 32-bit general purpose register.
5800 *
5801 * @returns Register reference.
5802 * @param pVCpu The cross context virtual CPU structure of the calling thread.
5803 * @param iReg The register.
5804 */
5805DECLINLINE(uint32_t *) iemGRegRefU32(PVMCPU pVCpu, uint8_t iReg)
5806{
5807 Assert(iReg < 16);
5808 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
5809 return &pCtx->aGRegs[iReg].u32;
5810}
5811
5812
5813/**
5814 * Gets a reference (pointer) to the specified 64-bit general purpose register.
5815 *
5816 * @returns Register reference.
5817 * @param pVCpu The cross context virtual CPU structure of the calling thread.
5818 * @param iReg The register.
5819 */
5820DECLINLINE(uint64_t *) iemGRegRefU64(PVMCPU pVCpu, uint8_t iReg)
5821{
5822 Assert(iReg < 64);
5823 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
5824 return &pCtx->aGRegs[iReg].u64;
5825}
5826
5827
5828/**
5829 * Fetches the value of a 8-bit general purpose register.
5830 *
5831 * @returns The register value.
5832 * @param pVCpu The cross context virtual CPU structure of the calling thread.
5833 * @param iReg The register.
5834 */
5835DECLINLINE(uint8_t) iemGRegFetchU8(PVMCPU pVCpu, uint8_t iReg)
5836{
5837 return *iemGRegRefU8(pVCpu, iReg);
5838}
5839
5840
5841/**
5842 * Fetches the value of a 16-bit general purpose register.
5843 *
5844 * @returns The register value.
5845 * @param pVCpu The cross context virtual CPU structure of the calling thread.
5846 * @param iReg The register.
5847 */
5848DECLINLINE(uint16_t) iemGRegFetchU16(PVMCPU pVCpu, uint8_t iReg)
5849{
5850 Assert(iReg < 16);
5851 return IEM_GET_CTX(pVCpu)->aGRegs[iReg].u16;
5852}
5853
5854
5855/**
5856 * Fetches the value of a 32-bit general purpose register.
5857 *
5858 * @returns The register value.
5859 * @param pVCpu The cross context virtual CPU structure of the calling thread.
5860 * @param iReg The register.
5861 */
5862DECLINLINE(uint32_t) iemGRegFetchU32(PVMCPU pVCpu, uint8_t iReg)
5863{
5864 Assert(iReg < 16);
5865 return IEM_GET_CTX(pVCpu)->aGRegs[iReg].u32;
5866}
5867
5868
5869/**
5870 * Fetches the value of a 64-bit general purpose register.
5871 *
5872 * @returns The register value.
5873 * @param pVCpu The cross context virtual CPU structure of the calling thread.
5874 * @param iReg The register.
5875 */
5876DECLINLINE(uint64_t) iemGRegFetchU64(PVMCPU pVCpu, uint8_t iReg)
5877{
5878 Assert(iReg < 16);
5879 return IEM_GET_CTX(pVCpu)->aGRegs[iReg].u64;
5880}
5881
5882
5883/**
5884 * Adds a 8-bit signed jump offset to RIP/EIP/IP.
5885 *
5886 * May raise a \#GP(0) if the new RIP is non-canonical or outside the code
5887 * segment limit.
5888 *
5889 * @param pVCpu The cross context virtual CPU structure of the calling thread.
5890 * @param offNextInstr The offset of the next instruction.
5891 */
5892IEM_STATIC VBOXSTRICTRC iemRegRipRelativeJumpS8(PVMCPU pVCpu, int8_t offNextInstr)
5893{
5894 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
5895 switch (pVCpu->iem.s.enmEffOpSize)
5896 {
5897 case IEMMODE_16BIT:
5898 {
5899 uint16_t uNewIp = pCtx->ip + offNextInstr + IEM_GET_INSTR_LEN(pVCpu);
5900 if ( uNewIp > pCtx->cs.u32Limit
5901 && pVCpu->iem.s.enmCpuMode != IEMMODE_64BIT) /* no need to check for non-canonical. */
5902 return iemRaiseGeneralProtectionFault0(pVCpu);
5903 pCtx->rip = uNewIp;
5904 break;
5905 }
5906
5907 case IEMMODE_32BIT:
5908 {
5909 Assert(pCtx->rip <= UINT32_MAX);
5910 Assert(pVCpu->iem.s.enmCpuMode != IEMMODE_64BIT);
5911
5912 uint32_t uNewEip = pCtx->eip + offNextInstr + IEM_GET_INSTR_LEN(pVCpu);
5913 if (uNewEip > pCtx->cs.u32Limit)
5914 return iemRaiseGeneralProtectionFault0(pVCpu);
5915 pCtx->rip = uNewEip;
5916 break;
5917 }
5918
5919 case IEMMODE_64BIT:
5920 {
5921 Assert(pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT);
5922
5923 uint64_t uNewRip = pCtx->rip + offNextInstr + IEM_GET_INSTR_LEN(pVCpu);
5924 if (!IEM_IS_CANONICAL(uNewRip))
5925 return iemRaiseGeneralProtectionFault0(pVCpu);
5926 pCtx->rip = uNewRip;
5927 break;
5928 }
5929
5930 IEM_NOT_REACHED_DEFAULT_CASE_RET();
5931 }
5932
5933 pCtx->eflags.Bits.u1RF = 0;
5934
5935#ifndef IEM_WITH_CODE_TLB
5936 /* Flush the prefetch buffer. */
5937 pVCpu->iem.s.cbOpcode = IEM_GET_INSTR_LEN(pVCpu);
5938#endif
5939
5940 return VINF_SUCCESS;
5941}
5942
5943
5944/**
5945 * Adds a 16-bit signed jump offset to RIP/EIP/IP.
5946 *
5947 * May raise a \#GP(0) if the new RIP is non-canonical or outside the code
5948 * segment limit.
5949 *
5950 * @returns Strict VBox status code.
5951 * @param pVCpu The cross context virtual CPU structure of the calling thread.
5952 * @param offNextInstr The offset of the next instruction.
5953 */
5954IEM_STATIC VBOXSTRICTRC iemRegRipRelativeJumpS16(PVMCPU pVCpu, int16_t offNextInstr)
5955{
5956 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
5957 Assert(pVCpu->iem.s.enmEffOpSize == IEMMODE_16BIT);
5958
5959 uint16_t uNewIp = pCtx->ip + offNextInstr + IEM_GET_INSTR_LEN(pVCpu);
5960 if ( uNewIp > pCtx->cs.u32Limit
5961 && pVCpu->iem.s.enmCpuMode != IEMMODE_64BIT) /* no need to check for non-canonical. */
5962 return iemRaiseGeneralProtectionFault0(pVCpu);
5963 /** @todo Test 16-bit jump in 64-bit mode. possible? */
5964 pCtx->rip = uNewIp;
5965 pCtx->eflags.Bits.u1RF = 0;
5966
5967#ifndef IEM_WITH_CODE_TLB
5968 /* Flush the prefetch buffer. */
5969 pVCpu->iem.s.cbOpcode = IEM_GET_INSTR_LEN(pVCpu);
5970#endif
5971
5972 return VINF_SUCCESS;
5973}
5974
5975
5976/**
5977 * Adds a 32-bit signed jump offset to RIP/EIP/IP.
5978 *
5979 * May raise a \#GP(0) if the new RIP is non-canonical or outside the code
5980 * segment limit.
5981 *
5982 * @returns Strict VBox status code.
5983 * @param pVCpu The cross context virtual CPU structure of the calling thread.
5984 * @param offNextInstr The offset of the next instruction.
5985 */
5986IEM_STATIC VBOXSTRICTRC iemRegRipRelativeJumpS32(PVMCPU pVCpu, int32_t offNextInstr)
5987{
5988 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
5989 Assert(pVCpu->iem.s.enmEffOpSize != IEMMODE_16BIT);
5990
5991 if (pVCpu->iem.s.enmEffOpSize == IEMMODE_32BIT)
5992 {
5993 Assert(pCtx->rip <= UINT32_MAX); Assert(pVCpu->iem.s.enmCpuMode != IEMMODE_64BIT);
5994
5995 uint32_t uNewEip = pCtx->eip + offNextInstr + IEM_GET_INSTR_LEN(pVCpu);
5996 if (uNewEip > pCtx->cs.u32Limit)
5997 return iemRaiseGeneralProtectionFault0(pVCpu);
5998 pCtx->rip = uNewEip;
5999 }
6000 else
6001 {
6002 Assert(pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT);
6003
6004 uint64_t uNewRip = pCtx->rip + offNextInstr + IEM_GET_INSTR_LEN(pVCpu);
6005 if (!IEM_IS_CANONICAL(uNewRip))
6006 return iemRaiseGeneralProtectionFault0(pVCpu);
6007 pCtx->rip = uNewRip;
6008 }
6009 pCtx->eflags.Bits.u1RF = 0;
6010
6011#ifndef IEM_WITH_CODE_TLB
6012 /* Flush the prefetch buffer. */
6013 pVCpu->iem.s.cbOpcode = IEM_GET_INSTR_LEN(pVCpu);
6014#endif
6015
6016 return VINF_SUCCESS;
6017}
6018
6019
6020/**
6021 * Performs a near jump to the specified address.
6022 *
6023 * May raise a \#GP(0) if the new RIP is non-canonical or outside the code
6024 * segment limit.
6025 *
6026 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6027 * @param uNewRip The new RIP value.
6028 */
6029IEM_STATIC VBOXSTRICTRC iemRegRipJump(PVMCPU pVCpu, uint64_t uNewRip)
6030{
6031 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
6032 switch (pVCpu->iem.s.enmEffOpSize)
6033 {
6034 case IEMMODE_16BIT:
6035 {
6036 Assert(uNewRip <= UINT16_MAX);
6037 if ( uNewRip > pCtx->cs.u32Limit
6038 && pVCpu->iem.s.enmCpuMode != IEMMODE_64BIT) /* no need to check for non-canonical. */
6039 return iemRaiseGeneralProtectionFault0(pVCpu);
6040 /** @todo Test 16-bit jump in 64-bit mode. */
6041 pCtx->rip = uNewRip;
6042 break;
6043 }
6044
6045 case IEMMODE_32BIT:
6046 {
6047 Assert(uNewRip <= UINT32_MAX);
6048 Assert(pCtx->rip <= UINT32_MAX);
6049 Assert(pVCpu->iem.s.enmCpuMode != IEMMODE_64BIT);
6050
6051 if (uNewRip > pCtx->cs.u32Limit)
6052 return iemRaiseGeneralProtectionFault0(pVCpu);
6053 pCtx->rip = uNewRip;
6054 break;
6055 }
6056
6057 case IEMMODE_64BIT:
6058 {
6059 Assert(pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT);
6060
6061 if (!IEM_IS_CANONICAL(uNewRip))
6062 return iemRaiseGeneralProtectionFault0(pVCpu);
6063 pCtx->rip = uNewRip;
6064 break;
6065 }
6066
6067 IEM_NOT_REACHED_DEFAULT_CASE_RET();
6068 }
6069
6070 pCtx->eflags.Bits.u1RF = 0;
6071
6072#ifndef IEM_WITH_CODE_TLB
6073 /* Flush the prefetch buffer. */
6074 pVCpu->iem.s.cbOpcode = IEM_GET_INSTR_LEN(pVCpu);
6075#endif
6076
6077 return VINF_SUCCESS;
6078}
6079
6080
6081/**
6082 * Get the address of the top of the stack.
6083 *
6084 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6085 * @param pCtx The CPU context which SP/ESP/RSP should be
6086 * read.
6087 */
6088DECLINLINE(RTGCPTR) iemRegGetEffRsp(PCVMCPU pVCpu, PCCPUMCTX pCtx)
6089{
6090 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
6091 return pCtx->rsp;
6092 if (pCtx->ss.Attr.n.u1DefBig)
6093 return pCtx->esp;
6094 return pCtx->sp;
6095}
6096
6097
6098/**
6099 * Updates the RIP/EIP/IP to point to the next instruction.
6100 *
6101 * This function leaves the EFLAGS.RF flag alone.
6102 *
6103 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6104 * @param cbInstr The number of bytes to add.
6105 */
6106IEM_STATIC void iemRegAddToRipKeepRF(PVMCPU pVCpu, uint8_t cbInstr)
6107{
6108 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
6109 switch (pVCpu->iem.s.enmCpuMode)
6110 {
6111 case IEMMODE_16BIT:
6112 Assert(pCtx->rip <= UINT16_MAX);
6113 pCtx->eip += cbInstr;
6114 pCtx->eip &= UINT32_C(0xffff);
6115 break;
6116
6117 case IEMMODE_32BIT:
6118 pCtx->eip += cbInstr;
6119 Assert(pCtx->rip <= UINT32_MAX);
6120 break;
6121
6122 case IEMMODE_64BIT:
6123 pCtx->rip += cbInstr;
6124 break;
6125 default: AssertFailed();
6126 }
6127}
6128
6129
6130#if 0
6131/**
6132 * Updates the RIP/EIP/IP to point to the next instruction.
6133 *
6134 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6135 */
6136IEM_STATIC void iemRegUpdateRipKeepRF(PVMCPU pVCpu)
6137{
6138 return iemRegAddToRipKeepRF(pVCpu, IEM_GET_INSTR_LEN(pVCpu));
6139}
6140#endif
6141
6142
6143
6144/**
6145 * Updates the RIP/EIP/IP to point to the next instruction and clears EFLAGS.RF.
6146 *
6147 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6148 * @param cbInstr The number of bytes to add.
6149 */
6150IEM_STATIC void iemRegAddToRipAndClearRF(PVMCPU pVCpu, uint8_t cbInstr)
6151{
6152 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
6153
6154 pCtx->eflags.Bits.u1RF = 0;
6155
6156 AssertCompile(IEMMODE_16BIT == 0 && IEMMODE_32BIT == 1 && IEMMODE_64BIT == 2);
6157#if ARCH_BITS >= 64
6158 static uint64_t const s_aRipMasks[] = { UINT64_C(0xffff), UINT64_C(0xffffffff), UINT64_MAX };
6159 Assert(pCtx->rip <= s_aRipMasks[(unsigned)pVCpu->iem.s.enmCpuMode]);
6160 pCtx->rip = (pCtx->rip + cbInstr) & s_aRipMasks[(unsigned)pVCpu->iem.s.enmCpuMode];
6161#else
6162 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
6163 pCtx->rip += cbInstr;
6164 else
6165 {
6166 static uint32_t const s_aEipMasks[] = { UINT32_C(0xffff), UINT32_MAX };
6167 pCtx->eip = (pCtx->eip + cbInstr) & s_aEipMasks[(unsigned)pVCpu->iem.s.enmCpuMode];
6168 }
6169#endif
6170}
6171
6172
6173/**
6174 * Updates the RIP/EIP/IP to point to the next instruction and clears EFLAGS.RF.
6175 *
6176 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6177 */
6178IEM_STATIC void iemRegUpdateRipAndClearRF(PVMCPU pVCpu)
6179{
6180 return iemRegAddToRipAndClearRF(pVCpu, IEM_GET_INSTR_LEN(pVCpu));
6181}
6182
6183
6184/**
6185 * Adds to the stack pointer.
6186 *
6187 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6188 * @param pCtx The CPU context which SP/ESP/RSP should be
6189 * updated.
6190 * @param cbToAdd The number of bytes to add (8-bit!).
6191 */
6192DECLINLINE(void) iemRegAddToRsp(PCVMCPU pVCpu, PCPUMCTX pCtx, uint8_t cbToAdd)
6193{
6194 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
6195 pCtx->rsp += cbToAdd;
6196 else if (pCtx->ss.Attr.n.u1DefBig)
6197 pCtx->esp += cbToAdd;
6198 else
6199 pCtx->sp += cbToAdd;
6200}
6201
6202
6203/**
6204 * Subtracts from the stack pointer.
6205 *
6206 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6207 * @param pCtx The CPU context which SP/ESP/RSP should be
6208 * updated.
6209 * @param cbToSub The number of bytes to subtract (8-bit!).
6210 */
6211DECLINLINE(void) iemRegSubFromRsp(PCVMCPU pVCpu, PCPUMCTX pCtx, uint8_t cbToSub)
6212{
6213 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
6214 pCtx->rsp -= cbToSub;
6215 else if (pCtx->ss.Attr.n.u1DefBig)
6216 pCtx->esp -= cbToSub;
6217 else
6218 pCtx->sp -= cbToSub;
6219}
6220
6221
6222/**
6223 * Adds to the temporary stack pointer.
6224 *
6225 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6226 * @param pTmpRsp The temporary SP/ESP/RSP to update.
6227 * @param cbToAdd The number of bytes to add (16-bit).
6228 * @param pCtx Where to get the current stack mode.
6229 */
6230DECLINLINE(void) iemRegAddToRspEx(PCVMCPU pVCpu, PCCPUMCTX pCtx, PRTUINT64U pTmpRsp, uint16_t cbToAdd)
6231{
6232 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
6233 pTmpRsp->u += cbToAdd;
6234 else if (pCtx->ss.Attr.n.u1DefBig)
6235 pTmpRsp->DWords.dw0 += cbToAdd;
6236 else
6237 pTmpRsp->Words.w0 += cbToAdd;
6238}
6239
6240
6241/**
6242 * Subtracts from the temporary stack pointer.
6243 *
6244 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6245 * @param pTmpRsp The temporary SP/ESP/RSP to update.
6246 * @param cbToSub The number of bytes to subtract.
6247 * @param pCtx Where to get the current stack mode.
6248 * @remarks The @a cbToSub argument *MUST* be 16-bit, iemCImpl_enter is
6249 * expecting that.
6250 */
6251DECLINLINE(void) iemRegSubFromRspEx(PCVMCPU pVCpu, PCCPUMCTX pCtx, PRTUINT64U pTmpRsp, uint16_t cbToSub)
6252{
6253 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
6254 pTmpRsp->u -= cbToSub;
6255 else if (pCtx->ss.Attr.n.u1DefBig)
6256 pTmpRsp->DWords.dw0 -= cbToSub;
6257 else
6258 pTmpRsp->Words.w0 -= cbToSub;
6259}
6260
6261
6262/**
6263 * Calculates the effective stack address for a push of the specified size as
6264 * well as the new RSP value (upper bits may be masked).
6265 *
6266 * @returns Effective stack addressf for the push.
6267 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6268 * @param pCtx Where to get the current stack mode.
6269 * @param cbItem The size of the stack item to pop.
6270 * @param puNewRsp Where to return the new RSP value.
6271 */
6272DECLINLINE(RTGCPTR) iemRegGetRspForPush(PCVMCPU pVCpu, PCCPUMCTX pCtx, uint8_t cbItem, uint64_t *puNewRsp)
6273{
6274 RTUINT64U uTmpRsp;
6275 RTGCPTR GCPtrTop;
6276 uTmpRsp.u = pCtx->rsp;
6277
6278 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
6279 GCPtrTop = uTmpRsp.u -= cbItem;
6280 else if (pCtx->ss.Attr.n.u1DefBig)
6281 GCPtrTop = uTmpRsp.DWords.dw0 -= cbItem;
6282 else
6283 GCPtrTop = uTmpRsp.Words.w0 -= cbItem;
6284 *puNewRsp = uTmpRsp.u;
6285 return GCPtrTop;
6286}
6287
6288
6289/**
6290 * Gets the current stack pointer and calculates the value after a pop of the
6291 * specified size.
6292 *
6293 * @returns Current stack pointer.
6294 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6295 * @param pCtx Where to get the current stack mode.
6296 * @param cbItem The size of the stack item to pop.
6297 * @param puNewRsp Where to return the new RSP value.
6298 */
6299DECLINLINE(RTGCPTR) iemRegGetRspForPop(PCVMCPU pVCpu, PCCPUMCTX pCtx, uint8_t cbItem, uint64_t *puNewRsp)
6300{
6301 RTUINT64U uTmpRsp;
6302 RTGCPTR GCPtrTop;
6303 uTmpRsp.u = pCtx->rsp;
6304
6305 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
6306 {
6307 GCPtrTop = uTmpRsp.u;
6308 uTmpRsp.u += cbItem;
6309 }
6310 else if (pCtx->ss.Attr.n.u1DefBig)
6311 {
6312 GCPtrTop = uTmpRsp.DWords.dw0;
6313 uTmpRsp.DWords.dw0 += cbItem;
6314 }
6315 else
6316 {
6317 GCPtrTop = uTmpRsp.Words.w0;
6318 uTmpRsp.Words.w0 += cbItem;
6319 }
6320 *puNewRsp = uTmpRsp.u;
6321 return GCPtrTop;
6322}
6323
6324
6325/**
6326 * Calculates the effective stack address for a push of the specified size as
6327 * well as the new temporary RSP value (upper bits may be masked).
6328 *
6329 * @returns Effective stack addressf for the push.
6330 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6331 * @param pCtx Where to get the current stack mode.
6332 * @param pTmpRsp The temporary stack pointer. This is updated.
6333 * @param cbItem The size of the stack item to pop.
6334 */
6335DECLINLINE(RTGCPTR) iemRegGetRspForPushEx(PCVMCPU pVCpu, PCCPUMCTX pCtx, PRTUINT64U pTmpRsp, uint8_t cbItem)
6336{
6337 RTGCPTR GCPtrTop;
6338
6339 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
6340 GCPtrTop = pTmpRsp->u -= cbItem;
6341 else if (pCtx->ss.Attr.n.u1DefBig)
6342 GCPtrTop = pTmpRsp->DWords.dw0 -= cbItem;
6343 else
6344 GCPtrTop = pTmpRsp->Words.w0 -= cbItem;
6345 return GCPtrTop;
6346}
6347
6348
6349/**
6350 * Gets the effective stack address for a pop of the specified size and
6351 * calculates and updates the temporary RSP.
6352 *
6353 * @returns Current stack pointer.
6354 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6355 * @param pCtx Where to get the current stack mode.
6356 * @param pTmpRsp The temporary stack pointer. This is updated.
6357 * @param cbItem The size of the stack item to pop.
6358 */
6359DECLINLINE(RTGCPTR) iemRegGetRspForPopEx(PCVMCPU pVCpu, PCCPUMCTX pCtx, PRTUINT64U pTmpRsp, uint8_t cbItem)
6360{
6361 RTGCPTR GCPtrTop;
6362 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
6363 {
6364 GCPtrTop = pTmpRsp->u;
6365 pTmpRsp->u += cbItem;
6366 }
6367 else if (pCtx->ss.Attr.n.u1DefBig)
6368 {
6369 GCPtrTop = pTmpRsp->DWords.dw0;
6370 pTmpRsp->DWords.dw0 += cbItem;
6371 }
6372 else
6373 {
6374 GCPtrTop = pTmpRsp->Words.w0;
6375 pTmpRsp->Words.w0 += cbItem;
6376 }
6377 return GCPtrTop;
6378}
6379
6380/** @} */
6381
6382
6383/** @name FPU access and helpers.
6384 *
6385 * @{
6386 */
6387
6388
6389/**
6390 * Hook for preparing to use the host FPU.
6391 *
6392 * This is necessary in ring-0 and raw-mode context (nop in ring-3).
6393 *
6394 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6395 */
6396DECLINLINE(void) iemFpuPrepareUsage(PVMCPU pVCpu)
6397{
6398#ifdef IN_RING3
6399 CPUMSetChangedFlags(pVCpu, CPUM_CHANGED_FPU_REM);
6400#else
6401 CPUMRZFpuStatePrepareHostCpuForUse(pVCpu);
6402#endif
6403}
6404
6405
6406/**
6407 * Hook for preparing to use the host FPU for SSE
6408 *
6409 * This is necessary in ring-0 and raw-mode context (nop in ring-3).
6410 *
6411 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6412 */
6413DECLINLINE(void) iemFpuPrepareUsageSse(PVMCPU pVCpu)
6414{
6415 iemFpuPrepareUsage(pVCpu);
6416}
6417
6418
6419/**
6420 * Hook for actualizing the guest FPU state before the interpreter reads it.
6421 *
6422 * This is necessary in ring-0 and raw-mode context (nop in ring-3).
6423 *
6424 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6425 */
6426DECLINLINE(void) iemFpuActualizeStateForRead(PVMCPU pVCpu)
6427{
6428#ifdef IN_RING3
6429 NOREF(pVCpu);
6430#else
6431 CPUMRZFpuStateActualizeForRead(pVCpu);
6432#endif
6433}
6434
6435
6436/**
6437 * Hook for actualizing the guest FPU state before the interpreter changes it.
6438 *
6439 * This is necessary in ring-0 and raw-mode context (nop in ring-3).
6440 *
6441 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6442 */
6443DECLINLINE(void) iemFpuActualizeStateForChange(PVMCPU pVCpu)
6444{
6445#ifdef IN_RING3
6446 CPUMSetChangedFlags(pVCpu, CPUM_CHANGED_FPU_REM);
6447#else
6448 CPUMRZFpuStateActualizeForChange(pVCpu);
6449#endif
6450}
6451
6452
6453/**
6454 * Hook for actualizing the guest XMM0..15 register state for read only.
6455 *
6456 * This is necessary in ring-0 and raw-mode context (nop in ring-3).
6457 *
6458 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6459 */
6460DECLINLINE(void) iemFpuActualizeSseStateForRead(PVMCPU pVCpu)
6461{
6462#if defined(IN_RING3) || defined(VBOX_WITH_KERNEL_USING_XMM)
6463 NOREF(pVCpu);
6464#else
6465 CPUMRZFpuStateActualizeSseForRead(pVCpu);
6466#endif
6467}
6468
6469
6470/**
6471 * Hook for actualizing the guest XMM0..15 register state for read+write.
6472 *
6473 * This is necessary in ring-0 and raw-mode context (nop in ring-3).
6474 *
6475 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6476 */
6477DECLINLINE(void) iemFpuActualizeSseStateForChange(PVMCPU pVCpu)
6478{
6479#if defined(IN_RING3) || defined(VBOX_WITH_KERNEL_USING_XMM)
6480 CPUMSetChangedFlags(pVCpu, CPUM_CHANGED_FPU_REM);
6481#else
6482 CPUMRZFpuStateActualizeForChange(pVCpu);
6483#endif
6484}
6485
6486
6487/**
6488 * Stores a QNaN value into a FPU register.
6489 *
6490 * @param pReg Pointer to the register.
6491 */
6492DECLINLINE(void) iemFpuStoreQNan(PRTFLOAT80U pReg)
6493{
6494 pReg->au32[0] = UINT32_C(0x00000000);
6495 pReg->au32[1] = UINT32_C(0xc0000000);
6496 pReg->au16[4] = UINT16_C(0xffff);
6497}
6498
6499
6500/**
6501 * Updates the FOP, FPU.CS and FPUIP registers.
6502 *
6503 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6504 * @param pCtx The CPU context.
6505 * @param pFpuCtx The FPU context.
6506 */
6507DECLINLINE(void) iemFpuUpdateOpcodeAndIpWorker(PVMCPU pVCpu, PCPUMCTX pCtx, PX86FXSTATE pFpuCtx)
6508{
6509 Assert(pVCpu->iem.s.uFpuOpcode != UINT16_MAX);
6510 pFpuCtx->FOP = pVCpu->iem.s.uFpuOpcode;
6511 /** @todo x87.CS and FPUIP needs to be kept seperately. */
6512 if (IEM_IS_REAL_OR_V86_MODE(pVCpu))
6513 {
6514 /** @todo Testcase: making assumptions about how FPUIP and FPUDP are handled
6515 * happens in real mode here based on the fnsave and fnstenv images. */
6516 pFpuCtx->CS = 0;
6517 pFpuCtx->FPUIP = pCtx->eip | ((uint32_t)pCtx->cs.Sel << 4);
6518 }
6519 else
6520 {
6521 pFpuCtx->CS = pCtx->cs.Sel;
6522 pFpuCtx->FPUIP = pCtx->rip;
6523 }
6524}
6525
6526
6527/**
6528 * Updates the x87.DS and FPUDP registers.
6529 *
6530 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6531 * @param pCtx The CPU context.
6532 * @param pFpuCtx The FPU context.
6533 * @param iEffSeg The effective segment register.
6534 * @param GCPtrEff The effective address relative to @a iEffSeg.
6535 */
6536DECLINLINE(void) iemFpuUpdateDP(PVMCPU pVCpu, PCPUMCTX pCtx, PX86FXSTATE pFpuCtx, uint8_t iEffSeg, RTGCPTR GCPtrEff)
6537{
6538 RTSEL sel;
6539 switch (iEffSeg)
6540 {
6541 case X86_SREG_DS: sel = pCtx->ds.Sel; break;
6542 case X86_SREG_SS: sel = pCtx->ss.Sel; break;
6543 case X86_SREG_CS: sel = pCtx->cs.Sel; break;
6544 case X86_SREG_ES: sel = pCtx->es.Sel; break;
6545 case X86_SREG_FS: sel = pCtx->fs.Sel; break;
6546 case X86_SREG_GS: sel = pCtx->gs.Sel; break;
6547 default:
6548 AssertMsgFailed(("%d\n", iEffSeg));
6549 sel = pCtx->ds.Sel;
6550 }
6551 /** @todo pFpuCtx->DS and FPUDP needs to be kept seperately. */
6552 if (IEM_IS_REAL_OR_V86_MODE(pVCpu))
6553 {
6554 pFpuCtx->DS = 0;
6555 pFpuCtx->FPUDP = (uint32_t)GCPtrEff + ((uint32_t)sel << 4);
6556 }
6557 else
6558 {
6559 pFpuCtx->DS = sel;
6560 pFpuCtx->FPUDP = GCPtrEff;
6561 }
6562}
6563
6564
6565/**
6566 * Rotates the stack registers in the push direction.
6567 *
6568 * @param pFpuCtx The FPU context.
6569 * @remarks This is a complete waste of time, but fxsave stores the registers in
6570 * stack order.
6571 */
6572DECLINLINE(void) iemFpuRotateStackPush(PX86FXSTATE pFpuCtx)
6573{
6574 RTFLOAT80U r80Tmp = pFpuCtx->aRegs[7].r80;
6575 pFpuCtx->aRegs[7].r80 = pFpuCtx->aRegs[6].r80;
6576 pFpuCtx->aRegs[6].r80 = pFpuCtx->aRegs[5].r80;
6577 pFpuCtx->aRegs[5].r80 = pFpuCtx->aRegs[4].r80;
6578 pFpuCtx->aRegs[4].r80 = pFpuCtx->aRegs[3].r80;
6579 pFpuCtx->aRegs[3].r80 = pFpuCtx->aRegs[2].r80;
6580 pFpuCtx->aRegs[2].r80 = pFpuCtx->aRegs[1].r80;
6581 pFpuCtx->aRegs[1].r80 = pFpuCtx->aRegs[0].r80;
6582 pFpuCtx->aRegs[0].r80 = r80Tmp;
6583}
6584
6585
6586/**
6587 * Rotates the stack registers in the pop direction.
6588 *
6589 * @param pFpuCtx The FPU context.
6590 * @remarks This is a complete waste of time, but fxsave stores the registers in
6591 * stack order.
6592 */
6593DECLINLINE(void) iemFpuRotateStackPop(PX86FXSTATE pFpuCtx)
6594{
6595 RTFLOAT80U r80Tmp = pFpuCtx->aRegs[0].r80;
6596 pFpuCtx->aRegs[0].r80 = pFpuCtx->aRegs[1].r80;
6597 pFpuCtx->aRegs[1].r80 = pFpuCtx->aRegs[2].r80;
6598 pFpuCtx->aRegs[2].r80 = pFpuCtx->aRegs[3].r80;
6599 pFpuCtx->aRegs[3].r80 = pFpuCtx->aRegs[4].r80;
6600 pFpuCtx->aRegs[4].r80 = pFpuCtx->aRegs[5].r80;
6601 pFpuCtx->aRegs[5].r80 = pFpuCtx->aRegs[6].r80;
6602 pFpuCtx->aRegs[6].r80 = pFpuCtx->aRegs[7].r80;
6603 pFpuCtx->aRegs[7].r80 = r80Tmp;
6604}
6605
6606
6607/**
6608 * Updates FSW and pushes a FPU result onto the FPU stack if no pending
6609 * exception prevents it.
6610 *
6611 * @param pResult The FPU operation result to push.
6612 * @param pFpuCtx The FPU context.
6613 */
6614IEM_STATIC void iemFpuMaybePushResult(PIEMFPURESULT pResult, PX86FXSTATE pFpuCtx)
6615{
6616 /* Update FSW and bail if there are pending exceptions afterwards. */
6617 uint16_t fFsw = pFpuCtx->FSW & ~X86_FSW_C_MASK;
6618 fFsw |= pResult->FSW & ~X86_FSW_TOP_MASK;
6619 if ( (fFsw & (X86_FSW_IE | X86_FSW_ZE | X86_FSW_DE))
6620 & ~(pFpuCtx->FCW & (X86_FCW_IM | X86_FCW_ZM | X86_FCW_DM)))
6621 {
6622 pFpuCtx->FSW = fFsw;
6623 return;
6624 }
6625
6626 uint16_t iNewTop = (X86_FSW_TOP_GET(fFsw) + 7) & X86_FSW_TOP_SMASK;
6627 if (!(pFpuCtx->FTW & RT_BIT(iNewTop)))
6628 {
6629 /* All is fine, push the actual value. */
6630 pFpuCtx->FTW |= RT_BIT(iNewTop);
6631 pFpuCtx->aRegs[7].r80 = pResult->r80Result;
6632 }
6633 else if (pFpuCtx->FCW & X86_FCW_IM)
6634 {
6635 /* Masked stack overflow, push QNaN. */
6636 fFsw |= X86_FSW_IE | X86_FSW_SF | X86_FSW_C1;
6637 iemFpuStoreQNan(&pFpuCtx->aRegs[7].r80);
6638 }
6639 else
6640 {
6641 /* Raise stack overflow, don't push anything. */
6642 pFpuCtx->FSW |= pResult->FSW & ~X86_FSW_C_MASK;
6643 pFpuCtx->FSW |= X86_FSW_IE | X86_FSW_SF | X86_FSW_C1 | X86_FSW_B | X86_FSW_ES;
6644 return;
6645 }
6646
6647 fFsw &= ~X86_FSW_TOP_MASK;
6648 fFsw |= iNewTop << X86_FSW_TOP_SHIFT;
6649 pFpuCtx->FSW = fFsw;
6650
6651 iemFpuRotateStackPush(pFpuCtx);
6652}
6653
6654
6655/**
6656 * Stores a result in a FPU register and updates the FSW and FTW.
6657 *
6658 * @param pFpuCtx The FPU context.
6659 * @param pResult The result to store.
6660 * @param iStReg Which FPU register to store it in.
6661 */
6662IEM_STATIC void iemFpuStoreResultOnly(PX86FXSTATE pFpuCtx, PIEMFPURESULT pResult, uint8_t iStReg)
6663{
6664 Assert(iStReg < 8);
6665 uint16_t iReg = (X86_FSW_TOP_GET(pFpuCtx->FSW) + iStReg) & X86_FSW_TOP_SMASK;
6666 pFpuCtx->FSW &= ~X86_FSW_C_MASK;
6667 pFpuCtx->FSW |= pResult->FSW & ~X86_FSW_TOP_MASK;
6668 pFpuCtx->FTW |= RT_BIT(iReg);
6669 pFpuCtx->aRegs[iStReg].r80 = pResult->r80Result;
6670}
6671
6672
6673/**
6674 * Only updates the FPU status word (FSW) with the result of the current
6675 * instruction.
6676 *
6677 * @param pFpuCtx The FPU context.
6678 * @param u16FSW The FSW output of the current instruction.
6679 */
6680IEM_STATIC void iemFpuUpdateFSWOnly(PX86FXSTATE pFpuCtx, uint16_t u16FSW)
6681{
6682 pFpuCtx->FSW &= ~X86_FSW_C_MASK;
6683 pFpuCtx->FSW |= u16FSW & ~X86_FSW_TOP_MASK;
6684}
6685
6686
6687/**
6688 * Pops one item off the FPU stack if no pending exception prevents it.
6689 *
6690 * @param pFpuCtx The FPU context.
6691 */
6692IEM_STATIC void iemFpuMaybePopOne(PX86FXSTATE pFpuCtx)
6693{
6694 /* Check pending exceptions. */
6695 uint16_t uFSW = pFpuCtx->FSW;
6696 if ( (pFpuCtx->FSW & (X86_FSW_IE | X86_FSW_ZE | X86_FSW_DE))
6697 & ~(pFpuCtx->FCW & (X86_FCW_IM | X86_FCW_ZM | X86_FCW_DM)))
6698 return;
6699
6700 /* TOP--. */
6701 uint16_t iOldTop = uFSW & X86_FSW_TOP_MASK;
6702 uFSW &= ~X86_FSW_TOP_MASK;
6703 uFSW |= (iOldTop + (UINT16_C(9) << X86_FSW_TOP_SHIFT)) & X86_FSW_TOP_MASK;
6704 pFpuCtx->FSW = uFSW;
6705
6706 /* Mark the previous ST0 as empty. */
6707 iOldTop >>= X86_FSW_TOP_SHIFT;
6708 pFpuCtx->FTW &= ~RT_BIT(iOldTop);
6709
6710 /* Rotate the registers. */
6711 iemFpuRotateStackPop(pFpuCtx);
6712}
6713
6714
6715/**
6716 * Pushes a FPU result onto the FPU stack if no pending exception prevents it.
6717 *
6718 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6719 * @param pResult The FPU operation result to push.
6720 */
6721IEM_STATIC void iemFpuPushResult(PVMCPU pVCpu, PIEMFPURESULT pResult)
6722{
6723 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
6724 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
6725 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
6726 iemFpuMaybePushResult(pResult, pFpuCtx);
6727}
6728
6729
6730/**
6731 * Pushes a FPU result onto the FPU stack if no pending exception prevents it,
6732 * and sets FPUDP and FPUDS.
6733 *
6734 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6735 * @param pResult The FPU operation result to push.
6736 * @param iEffSeg The effective segment register.
6737 * @param GCPtrEff The effective address relative to @a iEffSeg.
6738 */
6739IEM_STATIC void iemFpuPushResultWithMemOp(PVMCPU pVCpu, PIEMFPURESULT pResult, uint8_t iEffSeg, RTGCPTR GCPtrEff)
6740{
6741 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
6742 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
6743 iemFpuUpdateDP(pVCpu, pCtx, pFpuCtx, iEffSeg, GCPtrEff);
6744 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
6745 iemFpuMaybePushResult(pResult, pFpuCtx);
6746}
6747
6748
6749/**
6750 * Replace ST0 with the first value and push the second onto the FPU stack,
6751 * unless a pending exception prevents it.
6752 *
6753 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6754 * @param pResult The FPU operation result to store and push.
6755 */
6756IEM_STATIC void iemFpuPushResultTwo(PVMCPU pVCpu, PIEMFPURESULTTWO pResult)
6757{
6758 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
6759 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
6760 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
6761
6762 /* Update FSW and bail if there are pending exceptions afterwards. */
6763 uint16_t fFsw = pFpuCtx->FSW & ~X86_FSW_C_MASK;
6764 fFsw |= pResult->FSW & ~X86_FSW_TOP_MASK;
6765 if ( (fFsw & (X86_FSW_IE | X86_FSW_ZE | X86_FSW_DE))
6766 & ~(pFpuCtx->FCW & (X86_FCW_IM | X86_FCW_ZM | X86_FCW_DM)))
6767 {
6768 pFpuCtx->FSW = fFsw;
6769 return;
6770 }
6771
6772 uint16_t iNewTop = (X86_FSW_TOP_GET(fFsw) + 7) & X86_FSW_TOP_SMASK;
6773 if (!(pFpuCtx->FTW & RT_BIT(iNewTop)))
6774 {
6775 /* All is fine, push the actual value. */
6776 pFpuCtx->FTW |= RT_BIT(iNewTop);
6777 pFpuCtx->aRegs[0].r80 = pResult->r80Result1;
6778 pFpuCtx->aRegs[7].r80 = pResult->r80Result2;
6779 }
6780 else if (pFpuCtx->FCW & X86_FCW_IM)
6781 {
6782 /* Masked stack overflow, push QNaN. */
6783 fFsw |= X86_FSW_IE | X86_FSW_SF | X86_FSW_C1;
6784 iemFpuStoreQNan(&pFpuCtx->aRegs[0].r80);
6785 iemFpuStoreQNan(&pFpuCtx->aRegs[7].r80);
6786 }
6787 else
6788 {
6789 /* Raise stack overflow, don't push anything. */
6790 pFpuCtx->FSW |= pResult->FSW & ~X86_FSW_C_MASK;
6791 pFpuCtx->FSW |= X86_FSW_IE | X86_FSW_SF | X86_FSW_C1 | X86_FSW_B | X86_FSW_ES;
6792 return;
6793 }
6794
6795 fFsw &= ~X86_FSW_TOP_MASK;
6796 fFsw |= iNewTop << X86_FSW_TOP_SHIFT;
6797 pFpuCtx->FSW = fFsw;
6798
6799 iemFpuRotateStackPush(pFpuCtx);
6800}
6801
6802
6803/**
6804 * Stores a result in a FPU register, updates the FSW, FTW, FPUIP, FPUCS, and
6805 * FOP.
6806 *
6807 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6808 * @param pResult The result to store.
6809 * @param iStReg Which FPU register to store it in.
6810 */
6811IEM_STATIC void iemFpuStoreResult(PVMCPU pVCpu, PIEMFPURESULT pResult, uint8_t iStReg)
6812{
6813 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
6814 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
6815 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
6816 iemFpuStoreResultOnly(pFpuCtx, pResult, iStReg);
6817}
6818
6819
6820/**
6821 * Stores a result in a FPU register, updates the FSW, FTW, FPUIP, FPUCS, and
6822 * FOP, and then pops the stack.
6823 *
6824 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6825 * @param pResult The result to store.
6826 * @param iStReg Which FPU register to store it in.
6827 */
6828IEM_STATIC void iemFpuStoreResultThenPop(PVMCPU pVCpu, PIEMFPURESULT pResult, uint8_t iStReg)
6829{
6830 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
6831 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
6832 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
6833 iemFpuStoreResultOnly(pFpuCtx, pResult, iStReg);
6834 iemFpuMaybePopOne(pFpuCtx);
6835}
6836
6837
6838/**
6839 * Stores a result in a FPU register, updates the FSW, FTW, FPUIP, FPUCS, FOP,
6840 * FPUDP, and FPUDS.
6841 *
6842 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6843 * @param pResult The result to store.
6844 * @param iStReg Which FPU register to store it in.
6845 * @param iEffSeg The effective memory operand selector register.
6846 * @param GCPtrEff The effective memory operand offset.
6847 */
6848IEM_STATIC void iemFpuStoreResultWithMemOp(PVMCPU pVCpu, PIEMFPURESULT pResult, uint8_t iStReg,
6849 uint8_t iEffSeg, RTGCPTR GCPtrEff)
6850{
6851 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
6852 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
6853 iemFpuUpdateDP(pVCpu, pCtx, pFpuCtx, iEffSeg, GCPtrEff);
6854 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
6855 iemFpuStoreResultOnly(pFpuCtx, pResult, iStReg);
6856}
6857
6858
6859/**
6860 * Stores a result in a FPU register, updates the FSW, FTW, FPUIP, FPUCS, FOP,
6861 * FPUDP, and FPUDS, and then pops the stack.
6862 *
6863 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6864 * @param pResult The result to store.
6865 * @param iStReg Which FPU register to store it in.
6866 * @param iEffSeg The effective memory operand selector register.
6867 * @param GCPtrEff The effective memory operand offset.
6868 */
6869IEM_STATIC void iemFpuStoreResultWithMemOpThenPop(PVMCPU pVCpu, PIEMFPURESULT pResult,
6870 uint8_t iStReg, uint8_t iEffSeg, RTGCPTR GCPtrEff)
6871{
6872 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
6873 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
6874 iemFpuUpdateDP(pVCpu, pCtx, pFpuCtx, iEffSeg, GCPtrEff);
6875 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
6876 iemFpuStoreResultOnly(pFpuCtx, pResult, iStReg);
6877 iemFpuMaybePopOne(pFpuCtx);
6878}
6879
6880
6881/**
6882 * Updates the FOP, FPUIP, and FPUCS. For FNOP.
6883 *
6884 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6885 */
6886IEM_STATIC void iemFpuUpdateOpcodeAndIp(PVMCPU pVCpu)
6887{
6888 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
6889 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
6890 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
6891}
6892
6893
6894/**
6895 * Marks the specified stack register as free (for FFREE).
6896 *
6897 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6898 * @param iStReg The register to free.
6899 */
6900IEM_STATIC void iemFpuStackFree(PVMCPU pVCpu, uint8_t iStReg)
6901{
6902 Assert(iStReg < 8);
6903 PX86FXSTATE pFpuCtx = &IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87;
6904 uint8_t iReg = (X86_FSW_TOP_GET(pFpuCtx->FSW) + iStReg) & X86_FSW_TOP_SMASK;
6905 pFpuCtx->FTW &= ~RT_BIT(iReg);
6906}
6907
6908
6909/**
6910 * Increments FSW.TOP, i.e. pops an item off the stack without freeing it.
6911 *
6912 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6913 */
6914IEM_STATIC void iemFpuStackIncTop(PVMCPU pVCpu)
6915{
6916 PX86FXSTATE pFpuCtx = &IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87;
6917 uint16_t uFsw = pFpuCtx->FSW;
6918 uint16_t uTop = uFsw & X86_FSW_TOP_MASK;
6919 uTop = (uTop + (1 << X86_FSW_TOP_SHIFT)) & X86_FSW_TOP_MASK;
6920 uFsw &= ~X86_FSW_TOP_MASK;
6921 uFsw |= uTop;
6922 pFpuCtx->FSW = uFsw;
6923}
6924
6925
6926/**
6927 * Decrements FSW.TOP, i.e. push an item off the stack without storing anything.
6928 *
6929 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6930 */
6931IEM_STATIC void iemFpuStackDecTop(PVMCPU pVCpu)
6932{
6933 PX86FXSTATE pFpuCtx = &IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87;
6934 uint16_t uFsw = pFpuCtx->FSW;
6935 uint16_t uTop = uFsw & X86_FSW_TOP_MASK;
6936 uTop = (uTop + (7 << X86_FSW_TOP_SHIFT)) & X86_FSW_TOP_MASK;
6937 uFsw &= ~X86_FSW_TOP_MASK;
6938 uFsw |= uTop;
6939 pFpuCtx->FSW = uFsw;
6940}
6941
6942
6943/**
6944 * Updates the FSW, FOP, FPUIP, and FPUCS.
6945 *
6946 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6947 * @param u16FSW The FSW from the current instruction.
6948 */
6949IEM_STATIC void iemFpuUpdateFSW(PVMCPU pVCpu, uint16_t u16FSW)
6950{
6951 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
6952 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
6953 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
6954 iemFpuUpdateFSWOnly(pFpuCtx, u16FSW);
6955}
6956
6957
6958/**
6959 * Updates the FSW, FOP, FPUIP, and FPUCS, then pops the stack.
6960 *
6961 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6962 * @param u16FSW The FSW from the current instruction.
6963 */
6964IEM_STATIC void iemFpuUpdateFSWThenPop(PVMCPU pVCpu, uint16_t u16FSW)
6965{
6966 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
6967 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
6968 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
6969 iemFpuUpdateFSWOnly(pFpuCtx, u16FSW);
6970 iemFpuMaybePopOne(pFpuCtx);
6971}
6972
6973
6974/**
6975 * Updates the FSW, FOP, FPUIP, FPUCS, FPUDP, and FPUDS.
6976 *
6977 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6978 * @param u16FSW The FSW from the current instruction.
6979 * @param iEffSeg The effective memory operand selector register.
6980 * @param GCPtrEff The effective memory operand offset.
6981 */
6982IEM_STATIC void iemFpuUpdateFSWWithMemOp(PVMCPU pVCpu, uint16_t u16FSW, uint8_t iEffSeg, RTGCPTR GCPtrEff)
6983{
6984 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
6985 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
6986 iemFpuUpdateDP(pVCpu, pCtx, pFpuCtx, iEffSeg, GCPtrEff);
6987 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
6988 iemFpuUpdateFSWOnly(pFpuCtx, u16FSW);
6989}
6990
6991
6992/**
6993 * Updates the FSW, FOP, FPUIP, and FPUCS, then pops the stack twice.
6994 *
6995 * @param pVCpu The cross context virtual CPU structure of the calling thread.
6996 * @param u16FSW The FSW from the current instruction.
6997 */
6998IEM_STATIC void iemFpuUpdateFSWThenPopPop(PVMCPU pVCpu, uint16_t u16FSW)
6999{
7000 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
7001 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
7002 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
7003 iemFpuUpdateFSWOnly(pFpuCtx, u16FSW);
7004 iemFpuMaybePopOne(pFpuCtx);
7005 iemFpuMaybePopOne(pFpuCtx);
7006}
7007
7008
7009/**
7010 * Updates the FSW, FOP, FPUIP, FPUCS, FPUDP, and FPUDS, then pops the stack.
7011 *
7012 * @param pVCpu The cross context virtual CPU structure of the calling thread.
7013 * @param u16FSW The FSW from the current instruction.
7014 * @param iEffSeg The effective memory operand selector register.
7015 * @param GCPtrEff The effective memory operand offset.
7016 */
7017IEM_STATIC void iemFpuUpdateFSWWithMemOpThenPop(PVMCPU pVCpu, uint16_t u16FSW, uint8_t iEffSeg, RTGCPTR GCPtrEff)
7018{
7019 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
7020 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
7021 iemFpuUpdateDP(pVCpu, pCtx, pFpuCtx, iEffSeg, GCPtrEff);
7022 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
7023 iemFpuUpdateFSWOnly(pFpuCtx, u16FSW);
7024 iemFpuMaybePopOne(pFpuCtx);
7025}
7026
7027
7028/**
7029 * Worker routine for raising an FPU stack underflow exception.
7030 *
7031 * @param pFpuCtx The FPU context.
7032 * @param iStReg The stack register being accessed.
7033 */
7034IEM_STATIC void iemFpuStackUnderflowOnly(PX86FXSTATE pFpuCtx, uint8_t iStReg)
7035{
7036 Assert(iStReg < 8 || iStReg == UINT8_MAX);
7037 if (pFpuCtx->FCW & X86_FCW_IM)
7038 {
7039 /* Masked underflow. */
7040 pFpuCtx->FSW &= ~X86_FSW_C_MASK;
7041 pFpuCtx->FSW |= X86_FSW_IE | X86_FSW_SF;
7042 uint16_t iReg = (X86_FSW_TOP_GET(pFpuCtx->FSW) + iStReg) & X86_FSW_TOP_SMASK;
7043 if (iStReg != UINT8_MAX)
7044 {
7045 pFpuCtx->FTW |= RT_BIT(iReg);
7046 iemFpuStoreQNan(&pFpuCtx->aRegs[iStReg].r80);
7047 }
7048 }
7049 else
7050 {
7051 pFpuCtx->FSW &= ~X86_FSW_C_MASK;
7052 pFpuCtx->FSW |= X86_FSW_IE | X86_FSW_SF | X86_FSW_ES | X86_FSW_B;
7053 }
7054}
7055
7056
7057/**
7058 * Raises a FPU stack underflow exception.
7059 *
7060 * @param pVCpu The cross context virtual CPU structure of the calling thread.
7061 * @param iStReg The destination register that should be loaded
7062 * with QNaN if \#IS is not masked. Specify
7063 * UINT8_MAX if none (like for fcom).
7064 */
7065DECL_NO_INLINE(IEM_STATIC, void) iemFpuStackUnderflow(PVMCPU pVCpu, uint8_t iStReg)
7066{
7067 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
7068 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
7069 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
7070 iemFpuStackUnderflowOnly(pFpuCtx, iStReg);
7071}
7072
7073
7074DECL_NO_INLINE(IEM_STATIC, void)
7075iemFpuStackUnderflowWithMemOp(PVMCPU pVCpu, uint8_t iStReg, uint8_t iEffSeg, RTGCPTR GCPtrEff)
7076{
7077 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
7078 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
7079 iemFpuUpdateDP(pVCpu, pCtx, pFpuCtx, iEffSeg, GCPtrEff);
7080 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
7081 iemFpuStackUnderflowOnly(pFpuCtx, iStReg);
7082}
7083
7084
7085DECL_NO_INLINE(IEM_STATIC, void) iemFpuStackUnderflowThenPop(PVMCPU pVCpu, uint8_t iStReg)
7086{
7087 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
7088 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
7089 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
7090 iemFpuStackUnderflowOnly(pFpuCtx, iStReg);
7091 iemFpuMaybePopOne(pFpuCtx);
7092}
7093
7094
7095DECL_NO_INLINE(IEM_STATIC, void)
7096iemFpuStackUnderflowWithMemOpThenPop(PVMCPU pVCpu, uint8_t iStReg, uint8_t iEffSeg, RTGCPTR GCPtrEff)
7097{
7098 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
7099 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
7100 iemFpuUpdateDP(pVCpu, pCtx, pFpuCtx, iEffSeg, GCPtrEff);
7101 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
7102 iemFpuStackUnderflowOnly(pFpuCtx, iStReg);
7103 iemFpuMaybePopOne(pFpuCtx);
7104}
7105
7106
7107DECL_NO_INLINE(IEM_STATIC, void) iemFpuStackUnderflowThenPopPop(PVMCPU pVCpu)
7108{
7109 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
7110 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
7111 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
7112 iemFpuStackUnderflowOnly(pFpuCtx, UINT8_MAX);
7113 iemFpuMaybePopOne(pFpuCtx);
7114 iemFpuMaybePopOne(pFpuCtx);
7115}
7116
7117
7118DECL_NO_INLINE(IEM_STATIC, void)
7119iemFpuStackPushUnderflow(PVMCPU pVCpu)
7120{
7121 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
7122 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
7123 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
7124
7125 if (pFpuCtx->FCW & X86_FCW_IM)
7126 {
7127 /* Masked overflow - Push QNaN. */
7128 uint16_t iNewTop = (X86_FSW_TOP_GET(pFpuCtx->FSW) + 7) & X86_FSW_TOP_SMASK;
7129 pFpuCtx->FSW &= ~(X86_FSW_TOP_MASK | X86_FSW_C_MASK);
7130 pFpuCtx->FSW |= X86_FSW_IE | X86_FSW_SF;
7131 pFpuCtx->FSW |= iNewTop << X86_FSW_TOP_SHIFT;
7132 pFpuCtx->FTW |= RT_BIT(iNewTop);
7133 iemFpuStoreQNan(&pFpuCtx->aRegs[7].r80);
7134 iemFpuRotateStackPush(pFpuCtx);
7135 }
7136 else
7137 {
7138 /* Exception pending - don't change TOP or the register stack. */
7139 pFpuCtx->FSW &= ~X86_FSW_C_MASK;
7140 pFpuCtx->FSW |= X86_FSW_IE | X86_FSW_SF | X86_FSW_ES | X86_FSW_B;
7141 }
7142}
7143
7144
7145DECL_NO_INLINE(IEM_STATIC, void)
7146iemFpuStackPushUnderflowTwo(PVMCPU pVCpu)
7147{
7148 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
7149 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
7150 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
7151
7152 if (pFpuCtx->FCW & X86_FCW_IM)
7153 {
7154 /* Masked overflow - Push QNaN. */
7155 uint16_t iNewTop = (X86_FSW_TOP_GET(pFpuCtx->FSW) + 7) & X86_FSW_TOP_SMASK;
7156 pFpuCtx->FSW &= ~(X86_FSW_TOP_MASK | X86_FSW_C_MASK);
7157 pFpuCtx->FSW |= X86_FSW_IE | X86_FSW_SF;
7158 pFpuCtx->FSW |= iNewTop << X86_FSW_TOP_SHIFT;
7159 pFpuCtx->FTW |= RT_BIT(iNewTop);
7160 iemFpuStoreQNan(&pFpuCtx->aRegs[0].r80);
7161 iemFpuStoreQNan(&pFpuCtx->aRegs[7].r80);
7162 iemFpuRotateStackPush(pFpuCtx);
7163 }
7164 else
7165 {
7166 /* Exception pending - don't change TOP or the register stack. */
7167 pFpuCtx->FSW &= ~X86_FSW_C_MASK;
7168 pFpuCtx->FSW |= X86_FSW_IE | X86_FSW_SF | X86_FSW_ES | X86_FSW_B;
7169 }
7170}
7171
7172
7173/**
7174 * Worker routine for raising an FPU stack overflow exception on a push.
7175 *
7176 * @param pFpuCtx The FPU context.
7177 */
7178IEM_STATIC void iemFpuStackPushOverflowOnly(PX86FXSTATE pFpuCtx)
7179{
7180 if (pFpuCtx->FCW & X86_FCW_IM)
7181 {
7182 /* Masked overflow. */
7183 uint16_t iNewTop = (X86_FSW_TOP_GET(pFpuCtx->FSW) + 7) & X86_FSW_TOP_SMASK;
7184 pFpuCtx->FSW &= ~(X86_FSW_TOP_MASK | X86_FSW_C_MASK);
7185 pFpuCtx->FSW |= X86_FSW_C1 | X86_FSW_IE | X86_FSW_SF;
7186 pFpuCtx->FSW |= iNewTop << X86_FSW_TOP_SHIFT;
7187 pFpuCtx->FTW |= RT_BIT(iNewTop);
7188 iemFpuStoreQNan(&pFpuCtx->aRegs[7].r80);
7189 iemFpuRotateStackPush(pFpuCtx);
7190 }
7191 else
7192 {
7193 /* Exception pending - don't change TOP or the register stack. */
7194 pFpuCtx->FSW &= ~X86_FSW_C_MASK;
7195 pFpuCtx->FSW |= X86_FSW_C1 | X86_FSW_IE | X86_FSW_SF | X86_FSW_ES | X86_FSW_B;
7196 }
7197}
7198
7199
7200/**
7201 * Raises a FPU stack overflow exception on a push.
7202 *
7203 * @param pVCpu The cross context virtual CPU structure of the calling thread.
7204 */
7205DECL_NO_INLINE(IEM_STATIC, void) iemFpuStackPushOverflow(PVMCPU pVCpu)
7206{
7207 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
7208 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
7209 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
7210 iemFpuStackPushOverflowOnly(pFpuCtx);
7211}
7212
7213
7214/**
7215 * Raises a FPU stack overflow exception on a push with a memory operand.
7216 *
7217 * @param pVCpu The cross context virtual CPU structure of the calling thread.
7218 * @param iEffSeg The effective memory operand selector register.
7219 * @param GCPtrEff The effective memory operand offset.
7220 */
7221DECL_NO_INLINE(IEM_STATIC, void)
7222iemFpuStackPushOverflowWithMemOp(PVMCPU pVCpu, uint8_t iEffSeg, RTGCPTR GCPtrEff)
7223{
7224 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
7225 PX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
7226 iemFpuUpdateDP(pVCpu, pCtx, pFpuCtx, iEffSeg, GCPtrEff);
7227 iemFpuUpdateOpcodeAndIpWorker(pVCpu, pCtx, pFpuCtx);
7228 iemFpuStackPushOverflowOnly(pFpuCtx);
7229}
7230
7231
7232IEM_STATIC int iemFpuStRegNotEmpty(PVMCPU pVCpu, uint8_t iStReg)
7233{
7234 PX86FXSTATE pFpuCtx = &IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87;
7235 uint16_t iReg = (X86_FSW_TOP_GET(pFpuCtx->FSW) + iStReg) & X86_FSW_TOP_SMASK;
7236 if (pFpuCtx->FTW & RT_BIT(iReg))
7237 return VINF_SUCCESS;
7238 return VERR_NOT_FOUND;
7239}
7240
7241
7242IEM_STATIC int iemFpuStRegNotEmptyRef(PVMCPU pVCpu, uint8_t iStReg, PCRTFLOAT80U *ppRef)
7243{
7244 PX86FXSTATE pFpuCtx = &IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87;
7245 uint16_t iReg = (X86_FSW_TOP_GET(pFpuCtx->FSW) + iStReg) & X86_FSW_TOP_SMASK;
7246 if (pFpuCtx->FTW & RT_BIT(iReg))
7247 {
7248 *ppRef = &pFpuCtx->aRegs[iStReg].r80;
7249 return VINF_SUCCESS;
7250 }
7251 return VERR_NOT_FOUND;
7252}
7253
7254
7255IEM_STATIC int iemFpu2StRegsNotEmptyRef(PVMCPU pVCpu, uint8_t iStReg0, PCRTFLOAT80U *ppRef0,
7256 uint8_t iStReg1, PCRTFLOAT80U *ppRef1)
7257{
7258 PX86FXSTATE pFpuCtx = &IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87;
7259 uint16_t iTop = X86_FSW_TOP_GET(pFpuCtx->FSW);
7260 uint16_t iReg0 = (iTop + iStReg0) & X86_FSW_TOP_SMASK;
7261 uint16_t iReg1 = (iTop + iStReg1) & X86_FSW_TOP_SMASK;
7262 if ((pFpuCtx->FTW & (RT_BIT(iReg0) | RT_BIT(iReg1))) == (RT_BIT(iReg0) | RT_BIT(iReg1)))
7263 {
7264 *ppRef0 = &pFpuCtx->aRegs[iStReg0].r80;
7265 *ppRef1 = &pFpuCtx->aRegs[iStReg1].r80;
7266 return VINF_SUCCESS;
7267 }
7268 return VERR_NOT_FOUND;
7269}
7270
7271
7272IEM_STATIC int iemFpu2StRegsNotEmptyRefFirst(PVMCPU pVCpu, uint8_t iStReg0, PCRTFLOAT80U *ppRef0, uint8_t iStReg1)
7273{
7274 PX86FXSTATE pFpuCtx = &IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87;
7275 uint16_t iTop = X86_FSW_TOP_GET(pFpuCtx->FSW);
7276 uint16_t iReg0 = (iTop + iStReg0) & X86_FSW_TOP_SMASK;
7277 uint16_t iReg1 = (iTop + iStReg1) & X86_FSW_TOP_SMASK;
7278 if ((pFpuCtx->FTW & (RT_BIT(iReg0) | RT_BIT(iReg1))) == (RT_BIT(iReg0) | RT_BIT(iReg1)))
7279 {
7280 *ppRef0 = &pFpuCtx->aRegs[iStReg0].r80;
7281 return VINF_SUCCESS;
7282 }
7283 return VERR_NOT_FOUND;
7284}
7285
7286
7287/**
7288 * Updates the FPU exception status after FCW is changed.
7289 *
7290 * @param pFpuCtx The FPU context.
7291 */
7292IEM_STATIC void iemFpuRecalcExceptionStatus(PX86FXSTATE pFpuCtx)
7293{
7294 uint16_t u16Fsw = pFpuCtx->FSW;
7295 if ((u16Fsw & X86_FSW_XCPT_MASK) & ~(pFpuCtx->FCW & X86_FCW_XCPT_MASK))
7296 u16Fsw |= X86_FSW_ES | X86_FSW_B;
7297 else
7298 u16Fsw &= ~(X86_FSW_ES | X86_FSW_B);
7299 pFpuCtx->FSW = u16Fsw;
7300}
7301
7302
7303/**
7304 * Calculates the full FTW (FPU tag word) for use in FNSTENV and FNSAVE.
7305 *
7306 * @returns The full FTW.
7307 * @param pFpuCtx The FPU context.
7308 */
7309IEM_STATIC uint16_t iemFpuCalcFullFtw(PCX86FXSTATE pFpuCtx)
7310{
7311 uint8_t const u8Ftw = (uint8_t)pFpuCtx->FTW;
7312 uint16_t u16Ftw = 0;
7313 unsigned const iTop = X86_FSW_TOP_GET(pFpuCtx->FSW);
7314 for (unsigned iSt = 0; iSt < 8; iSt++)
7315 {
7316 unsigned const iReg = (iSt + iTop) & 7;
7317 if (!(u8Ftw & RT_BIT(iReg)))
7318 u16Ftw |= 3 << (iReg * 2); /* empty */
7319 else
7320 {
7321 uint16_t uTag;
7322 PCRTFLOAT80U const pr80Reg = &pFpuCtx->aRegs[iSt].r80;
7323 if (pr80Reg->s.uExponent == 0x7fff)
7324 uTag = 2; /* Exponent is all 1's => Special. */
7325 else if (pr80Reg->s.uExponent == 0x0000)
7326 {
7327 if (pr80Reg->s.u64Mantissa == 0x0000)
7328 uTag = 1; /* All bits are zero => Zero. */
7329 else
7330 uTag = 2; /* Must be special. */
7331 }
7332 else if (pr80Reg->s.u64Mantissa & RT_BIT_64(63)) /* The J bit. */
7333 uTag = 0; /* Valid. */
7334 else
7335 uTag = 2; /* Must be special. */
7336
7337 u16Ftw |= uTag << (iReg * 2); /* empty */
7338 }
7339 }
7340
7341 return u16Ftw;
7342}
7343
7344
7345/**
7346 * Converts a full FTW to a compressed one (for use in FLDENV and FRSTOR).
7347 *
7348 * @returns The compressed FTW.
7349 * @param u16FullFtw The full FTW to convert.
7350 */
7351IEM_STATIC uint16_t iemFpuCompressFtw(uint16_t u16FullFtw)
7352{
7353 uint8_t u8Ftw = 0;
7354 for (unsigned i = 0; i < 8; i++)
7355 {
7356 if ((u16FullFtw & 3) != 3 /*empty*/)
7357 u8Ftw |= RT_BIT(i);
7358 u16FullFtw >>= 2;
7359 }
7360
7361 return u8Ftw;
7362}
7363
7364/** @} */
7365
7366
7367/** @name Memory access.
7368 *
7369 * @{
7370 */
7371
7372
7373/**
7374 * Updates the IEMCPU::cbWritten counter if applicable.
7375 *
7376 * @param pVCpu The cross context virtual CPU structure of the calling thread.
7377 * @param fAccess The access being accounted for.
7378 * @param cbMem The access size.
7379 */
7380DECL_FORCE_INLINE(void) iemMemUpdateWrittenCounter(PVMCPU pVCpu, uint32_t fAccess, size_t cbMem)
7381{
7382 if ( (fAccess & (IEM_ACCESS_WHAT_MASK | IEM_ACCESS_TYPE_WRITE)) == (IEM_ACCESS_WHAT_STACK | IEM_ACCESS_TYPE_WRITE)
7383 || (fAccess & (IEM_ACCESS_WHAT_MASK | IEM_ACCESS_TYPE_WRITE)) == (IEM_ACCESS_WHAT_DATA | IEM_ACCESS_TYPE_WRITE) )
7384 pVCpu->iem.s.cbWritten += (uint32_t)cbMem;
7385}
7386
7387
7388/**
7389 * Checks if the given segment can be written to, raise the appropriate
7390 * exception if not.
7391 *
7392 * @returns VBox strict status code.
7393 *
7394 * @param pVCpu The cross context virtual CPU structure of the calling thread.
7395 * @param pHid Pointer to the hidden register.
7396 * @param iSegReg The register number.
7397 * @param pu64BaseAddr Where to return the base address to use for the
7398 * segment. (In 64-bit code it may differ from the
7399 * base in the hidden segment.)
7400 */
7401IEM_STATIC VBOXSTRICTRC
7402iemMemSegCheckWriteAccessEx(PVMCPU pVCpu, PCCPUMSELREGHID pHid, uint8_t iSegReg, uint64_t *pu64BaseAddr)
7403{
7404 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
7405 *pu64BaseAddr = iSegReg < X86_SREG_FS ? 0 : pHid->u64Base;
7406 else
7407 {
7408 if (!pHid->Attr.n.u1Present)
7409 return iemRaiseSelectorNotPresentBySegReg(pVCpu, iSegReg);
7410
7411 if ( ( (pHid->Attr.n.u4Type & X86_SEL_TYPE_CODE)
7412 || !(pHid->Attr.n.u4Type & X86_SEL_TYPE_WRITE) )
7413 && pVCpu->iem.s.enmCpuMode != IEMMODE_64BIT )
7414 return iemRaiseSelectorInvalidAccess(pVCpu, iSegReg, IEM_ACCESS_DATA_W);
7415 *pu64BaseAddr = pHid->u64Base;
7416 }
7417 return VINF_SUCCESS;
7418}
7419
7420
7421/**
7422 * Checks if the given segment can be read from, raise the appropriate
7423 * exception if not.
7424 *
7425 * @returns VBox strict status code.
7426 *
7427 * @param pVCpu The cross context virtual CPU structure of the calling thread.
7428 * @param pHid Pointer to the hidden register.
7429 * @param iSegReg The register number.
7430 * @param pu64BaseAddr Where to return the base address to use for the
7431 * segment. (In 64-bit code it may differ from the
7432 * base in the hidden segment.)
7433 */
7434IEM_STATIC VBOXSTRICTRC
7435iemMemSegCheckReadAccessEx(PVMCPU pVCpu, PCCPUMSELREGHID pHid, uint8_t iSegReg, uint64_t *pu64BaseAddr)
7436{
7437 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
7438 *pu64BaseAddr = iSegReg < X86_SREG_FS ? 0 : pHid->u64Base;
7439 else
7440 {
7441 if (!pHid->Attr.n.u1Present)
7442 return iemRaiseSelectorNotPresentBySegReg(pVCpu, iSegReg);
7443
7444 if ((pHid->Attr.n.u4Type & (X86_SEL_TYPE_CODE | X86_SEL_TYPE_READ)) == X86_SEL_TYPE_CODE)
7445 return iemRaiseSelectorInvalidAccess(pVCpu, iSegReg, IEM_ACCESS_DATA_R);
7446 *pu64BaseAddr = pHid->u64Base;
7447 }
7448 return VINF_SUCCESS;
7449}
7450
7451
7452/**
7453 * Applies the segment limit, base and attributes.
7454 *
7455 * This may raise a \#GP or \#SS.
7456 *
7457 * @returns VBox strict status code.
7458 *
7459 * @param pVCpu The cross context virtual CPU structure of the calling thread.
7460 * @param fAccess The kind of access which is being performed.
7461 * @param iSegReg The index of the segment register to apply.
7462 * This is UINT8_MAX if none (for IDT, GDT, LDT,
7463 * TSS, ++).
7464 * @param cbMem The access size.
7465 * @param pGCPtrMem Pointer to the guest memory address to apply
7466 * segmentation to. Input and output parameter.
7467 */
7468IEM_STATIC VBOXSTRICTRC
7469iemMemApplySegment(PVMCPU pVCpu, uint32_t fAccess, uint8_t iSegReg, size_t cbMem, PRTGCPTR pGCPtrMem)
7470{
7471 if (iSegReg == UINT8_MAX)
7472 return VINF_SUCCESS;
7473
7474 PCPUMSELREGHID pSel = iemSRegGetHid(pVCpu, iSegReg);
7475 switch (pVCpu->iem.s.enmCpuMode)
7476 {
7477 case IEMMODE_16BIT:
7478 case IEMMODE_32BIT:
7479 {
7480 RTGCPTR32 GCPtrFirst32 = (RTGCPTR32)*pGCPtrMem;
7481 RTGCPTR32 GCPtrLast32 = GCPtrFirst32 + (uint32_t)cbMem - 1;
7482
7483 if ( pSel->Attr.n.u1Present
7484 && !pSel->Attr.n.u1Unusable)
7485 {
7486 Assert(pSel->Attr.n.u1DescType);
7487 if (!(pSel->Attr.n.u4Type & X86_SEL_TYPE_CODE))
7488 {
7489 if ( (fAccess & IEM_ACCESS_TYPE_WRITE)
7490 && !(pSel->Attr.n.u4Type & X86_SEL_TYPE_WRITE) )
7491 return iemRaiseSelectorInvalidAccess(pVCpu, iSegReg, fAccess);
7492
7493 if (!IEM_IS_REAL_OR_V86_MODE(pVCpu))
7494 {
7495 /** @todo CPL check. */
7496 }
7497
7498 /*
7499 * There are two kinds of data selectors, normal and expand down.
7500 */
7501 if (!(pSel->Attr.n.u4Type & X86_SEL_TYPE_DOWN))
7502 {
7503 if ( GCPtrFirst32 > pSel->u32Limit
7504 || GCPtrLast32 > pSel->u32Limit) /* yes, in real mode too (since 80286). */
7505 return iemRaiseSelectorBounds(pVCpu, iSegReg, fAccess);
7506 }
7507 else
7508 {
7509 /*
7510 * The upper boundary is defined by the B bit, not the G bit!
7511 */
7512 if ( GCPtrFirst32 < pSel->u32Limit + UINT32_C(1)
7513 || GCPtrLast32 > (pSel->Attr.n.u1DefBig ? UINT32_MAX : UINT32_C(0xffff)))
7514 return iemRaiseSelectorBounds(pVCpu, iSegReg, fAccess);
7515 }
7516 *pGCPtrMem = GCPtrFirst32 += (uint32_t)pSel->u64Base;
7517 }
7518 else
7519 {
7520
7521 /*
7522 * Code selector and usually be used to read thru, writing is
7523 * only permitted in real and V8086 mode.
7524 */
7525 if ( ( (fAccess & IEM_ACCESS_TYPE_WRITE)
7526 || ( (fAccess & IEM_ACCESS_TYPE_READ)
7527 && !(pSel->Attr.n.u4Type & X86_SEL_TYPE_READ)) )
7528 && !IEM_IS_REAL_OR_V86_MODE(pVCpu) )
7529 return iemRaiseSelectorInvalidAccess(pVCpu, iSegReg, fAccess);
7530
7531 if ( GCPtrFirst32 > pSel->u32Limit
7532 || GCPtrLast32 > pSel->u32Limit) /* yes, in real mode too (since 80286). */
7533 return iemRaiseSelectorBounds(pVCpu, iSegReg, fAccess);
7534
7535 if (!IEM_IS_REAL_OR_V86_MODE(pVCpu))
7536 {
7537 /** @todo CPL check. */
7538 }
7539
7540 *pGCPtrMem = GCPtrFirst32 += (uint32_t)pSel->u64Base;
7541 }
7542 }
7543 else
7544 return iemRaiseGeneralProtectionFault0(pVCpu);
7545 return VINF_SUCCESS;
7546 }
7547
7548 case IEMMODE_64BIT:
7549 {
7550 RTGCPTR GCPtrMem = *pGCPtrMem;
7551 if (iSegReg == X86_SREG_GS || iSegReg == X86_SREG_FS)
7552 *pGCPtrMem = GCPtrMem + pSel->u64Base;
7553
7554 Assert(cbMem >= 1);
7555 if (RT_LIKELY(X86_IS_CANONICAL(GCPtrMem) && X86_IS_CANONICAL(GCPtrMem + cbMem - 1)))
7556 return VINF_SUCCESS;
7557 return iemRaiseGeneralProtectionFault0(pVCpu);
7558 }
7559
7560 default:
7561 AssertFailedReturn(VERR_IEM_IPE_7);
7562 }
7563}
7564
7565
7566/**
7567 * Translates a virtual address to a physical physical address and checks if we
7568 * can access the page as specified.
7569 *
7570 * @param pVCpu The cross context virtual CPU structure of the calling thread.
7571 * @param GCPtrMem The virtual address.
7572 * @param fAccess The intended access.
7573 * @param pGCPhysMem Where to return the physical address.
7574 */
7575IEM_STATIC VBOXSTRICTRC
7576iemMemPageTranslateAndCheckAccess(PVMCPU pVCpu, RTGCPTR GCPtrMem, uint32_t fAccess, PRTGCPHYS pGCPhysMem)
7577{
7578 /** @todo Need a different PGM interface here. We're currently using
7579 * generic / REM interfaces. this won't cut it for R0 & RC. */
7580 RTGCPHYS GCPhys;
7581 uint64_t fFlags;
7582 int rc = PGMGstGetPage(pVCpu, GCPtrMem, &fFlags, &GCPhys);
7583 if (RT_FAILURE(rc))
7584 {
7585 /** @todo Check unassigned memory in unpaged mode. */
7586 /** @todo Reserved bits in page tables. Requires new PGM interface. */
7587 *pGCPhysMem = NIL_RTGCPHYS;
7588 return iemRaisePageFault(pVCpu, GCPtrMem, fAccess, rc);
7589 }
7590
7591 /* If the page is writable and does not have the no-exec bit set, all
7592 access is allowed. Otherwise we'll have to check more carefully... */
7593 if ((fFlags & (X86_PTE_RW | X86_PTE_US | X86_PTE_PAE_NX)) != (X86_PTE_RW | X86_PTE_US))
7594 {
7595 /* Write to read only memory? */
7596 if ( (fAccess & IEM_ACCESS_TYPE_WRITE)
7597 && !(fFlags & X86_PTE_RW)
7598 && ( pVCpu->iem.s.uCpl != 0
7599 || (IEM_GET_CTX(pVCpu)->cr0 & X86_CR0_WP)))
7600 {
7601 Log(("iemMemPageTranslateAndCheckAccess: GCPtrMem=%RGv - read-only page -> #PF\n", GCPtrMem));
7602 *pGCPhysMem = NIL_RTGCPHYS;
7603 return iemRaisePageFault(pVCpu, GCPtrMem, fAccess & ~IEM_ACCESS_TYPE_READ, VERR_ACCESS_DENIED);
7604 }
7605
7606 /* Kernel memory accessed by userland? */
7607 if ( !(fFlags & X86_PTE_US)
7608 && pVCpu->iem.s.uCpl == 3
7609 && !(fAccess & IEM_ACCESS_WHAT_SYS))
7610 {
7611 Log(("iemMemPageTranslateAndCheckAccess: GCPtrMem=%RGv - user access to kernel page -> #PF\n", GCPtrMem));
7612 *pGCPhysMem = NIL_RTGCPHYS;
7613 return iemRaisePageFault(pVCpu, GCPtrMem, fAccess, VERR_ACCESS_DENIED);
7614 }
7615
7616 /* Executing non-executable memory? */
7617 if ( (fAccess & IEM_ACCESS_TYPE_EXEC)
7618 && (fFlags & X86_PTE_PAE_NX)
7619 && (IEM_GET_CTX(pVCpu)->msrEFER & MSR_K6_EFER_NXE) )
7620 {
7621 Log(("iemMemPageTranslateAndCheckAccess: GCPtrMem=%RGv - NX -> #PF\n", GCPtrMem));
7622 *pGCPhysMem = NIL_RTGCPHYS;
7623 return iemRaisePageFault(pVCpu, GCPtrMem, fAccess & ~(IEM_ACCESS_TYPE_READ | IEM_ACCESS_TYPE_WRITE),
7624 VERR_ACCESS_DENIED);
7625 }
7626 }
7627
7628 /*
7629 * Set the dirty / access flags.
7630 * ASSUMES this is set when the address is translated rather than on committ...
7631 */
7632 /** @todo testcase: check when A and D bits are actually set by the CPU. */
7633 uint32_t fAccessedDirty = fAccess & IEM_ACCESS_TYPE_WRITE ? X86_PTE_D | X86_PTE_A : X86_PTE_A;
7634 if ((fFlags & fAccessedDirty) != fAccessedDirty)
7635 {
7636 int rc2 = PGMGstModifyPage(pVCpu, GCPtrMem, 1, fAccessedDirty, ~(uint64_t)fAccessedDirty);
7637 AssertRC(rc2);
7638 }
7639
7640 GCPhys |= GCPtrMem & PAGE_OFFSET_MASK;
7641 *pGCPhysMem = GCPhys;
7642 return VINF_SUCCESS;
7643}
7644
7645
7646
7647/**
7648 * Maps a physical page.
7649 *
7650 * @returns VBox status code (see PGMR3PhysTlbGCPhys2Ptr).
7651 * @param pVCpu The cross context virtual CPU structure of the calling thread.
7652 * @param GCPhysMem The physical address.
7653 * @param fAccess The intended access.
7654 * @param ppvMem Where to return the mapping address.
7655 * @param pLock The PGM lock.
7656 */
7657IEM_STATIC int iemMemPageMap(PVMCPU pVCpu, RTGCPHYS GCPhysMem, uint32_t fAccess, void **ppvMem, PPGMPAGEMAPLOCK pLock)
7658{
7659#ifdef IEM_VERIFICATION_MODE_FULL
7660 /* Force the alternative path so we can ignore writes. */
7661 if ((fAccess & IEM_ACCESS_TYPE_WRITE) && !pVCpu->iem.s.fNoRem)
7662 {
7663 if (IEM_FULL_VERIFICATION_ENABLED(pVCpu))
7664 {
7665 int rc2 = PGMPhysIemQueryAccess(pVCpu->CTX_SUFF(pVM), GCPhysMem,
7666 RT_BOOL(fAccess & IEM_ACCESS_TYPE_WRITE), pVCpu->iem.s.fBypassHandlers);
7667 if (RT_FAILURE(rc2))
7668 pVCpu->iem.s.fProblematicMemory = true;
7669 }
7670 return VERR_PGM_PHYS_TLB_CATCH_ALL;
7671 }
7672#endif
7673#ifdef IEM_LOG_MEMORY_WRITES
7674 if (fAccess & IEM_ACCESS_TYPE_WRITE)
7675 return VERR_PGM_PHYS_TLB_CATCH_ALL;
7676#endif
7677#ifdef IEM_VERIFICATION_MODE_MINIMAL
7678 return VERR_PGM_PHYS_TLB_CATCH_ALL;
7679#endif
7680
7681 /** @todo This API may require some improving later. A private deal with PGM
7682 * regarding locking and unlocking needs to be struct. A couple of TLBs
7683 * living in PGM, but with publicly accessible inlined access methods
7684 * could perhaps be an even better solution. */
7685 int rc = PGMPhysIemGCPhys2Ptr(pVCpu->CTX_SUFF(pVM), pVCpu,
7686 GCPhysMem,
7687 RT_BOOL(fAccess & IEM_ACCESS_TYPE_WRITE),
7688 pVCpu->iem.s.fBypassHandlers,
7689 ppvMem,
7690 pLock);
7691 /*Log(("PGMPhysIemGCPhys2Ptr %Rrc pLock=%.*Rhxs\n", rc, sizeof(*pLock), pLock));*/
7692 AssertMsg(rc == VINF_SUCCESS || RT_FAILURE_NP(rc), ("%Rrc\n", rc));
7693
7694#ifdef IEM_VERIFICATION_MODE_FULL
7695 if (RT_FAILURE(rc) && IEM_FULL_VERIFICATION_ENABLED(pVCpu))
7696 pVCpu->iem.s.fProblematicMemory = true;
7697#endif
7698 return rc;
7699}
7700
7701
7702/**
7703 * Unmap a page previously mapped by iemMemPageMap.
7704 *
7705 * @param pVCpu The cross context virtual CPU structure of the calling thread.
7706 * @param GCPhysMem The physical address.
7707 * @param fAccess The intended access.
7708 * @param pvMem What iemMemPageMap returned.
7709 * @param pLock The PGM lock.
7710 */
7711DECLINLINE(void) iemMemPageUnmap(PVMCPU pVCpu, RTGCPHYS GCPhysMem, uint32_t fAccess, const void *pvMem, PPGMPAGEMAPLOCK pLock)
7712{
7713 NOREF(pVCpu);
7714 NOREF(GCPhysMem);
7715 NOREF(fAccess);
7716 NOREF(pvMem);
7717 PGMPhysReleasePageMappingLock(pVCpu->CTX_SUFF(pVM), pLock);
7718}
7719
7720
7721/**
7722 * Looks up a memory mapping entry.
7723 *
7724 * @returns The mapping index (positive) or VERR_NOT_FOUND (negative).
7725 * @param pVCpu The cross context virtual CPU structure of the calling thread.
7726 * @param pvMem The memory address.
7727 * @param fAccess The access to.
7728 */
7729DECLINLINE(int) iemMapLookup(PVMCPU pVCpu, void *pvMem, uint32_t fAccess)
7730{
7731 Assert(pVCpu->iem.s.cActiveMappings <= RT_ELEMENTS(pVCpu->iem.s.aMemMappings));
7732 fAccess &= IEM_ACCESS_WHAT_MASK | IEM_ACCESS_TYPE_MASK;
7733 if ( pVCpu->iem.s.aMemMappings[0].pv == pvMem
7734 && (pVCpu->iem.s.aMemMappings[0].fAccess & (IEM_ACCESS_WHAT_MASK | IEM_ACCESS_TYPE_MASK)) == fAccess)
7735 return 0;
7736 if ( pVCpu->iem.s.aMemMappings[1].pv == pvMem
7737 && (pVCpu->iem.s.aMemMappings[1].fAccess & (IEM_ACCESS_WHAT_MASK | IEM_ACCESS_TYPE_MASK)) == fAccess)
7738 return 1;
7739 if ( pVCpu->iem.s.aMemMappings[2].pv == pvMem
7740 && (pVCpu->iem.s.aMemMappings[2].fAccess & (IEM_ACCESS_WHAT_MASK | IEM_ACCESS_TYPE_MASK)) == fAccess)
7741 return 2;
7742 return VERR_NOT_FOUND;
7743}
7744
7745
7746/**
7747 * Finds a free memmap entry when using iNextMapping doesn't work.
7748 *
7749 * @returns Memory mapping index, 1024 on failure.
7750 * @param pVCpu The cross context virtual CPU structure of the calling thread.
7751 */
7752IEM_STATIC unsigned iemMemMapFindFree(PVMCPU pVCpu)
7753{
7754 /*
7755 * The easy case.
7756 */
7757 if (pVCpu->iem.s.cActiveMappings == 0)
7758 {
7759 pVCpu->iem.s.iNextMapping = 1;
7760 return 0;
7761 }
7762
7763 /* There should be enough mappings for all instructions. */
7764 AssertReturn(pVCpu->iem.s.cActiveMappings < RT_ELEMENTS(pVCpu->iem.s.aMemMappings), 1024);
7765
7766 for (unsigned i = 0; i < RT_ELEMENTS(pVCpu->iem.s.aMemMappings); i++)
7767 if (pVCpu->iem.s.aMemMappings[i].fAccess == IEM_ACCESS_INVALID)
7768 return i;
7769
7770 AssertFailedReturn(1024);
7771}
7772
7773
7774/**
7775 * Commits a bounce buffer that needs writing back and unmaps it.
7776 *
7777 * @returns Strict VBox status code.
7778 * @param pVCpu The cross context virtual CPU structure of the calling thread.
7779 * @param iMemMap The index of the buffer to commit.
7780 * @param fPostponeFail Whether we can postpone writer failures to ring-3.
7781 * Always false in ring-3, obviously.
7782 */
7783IEM_STATIC VBOXSTRICTRC iemMemBounceBufferCommitAndUnmap(PVMCPU pVCpu, unsigned iMemMap, bool fPostponeFail)
7784{
7785 Assert(pVCpu->iem.s.aMemMappings[iMemMap].fAccess & IEM_ACCESS_BOUNCE_BUFFERED);
7786 Assert(pVCpu->iem.s.aMemMappings[iMemMap].fAccess & IEM_ACCESS_TYPE_WRITE);
7787#ifdef IN_RING3
7788 Assert(!fPostponeFail);
7789 RT_NOREF_PV(fPostponeFail);
7790#endif
7791
7792 /*
7793 * Do the writing.
7794 */
7795#ifndef IEM_VERIFICATION_MODE_MINIMAL
7796 PVM pVM = pVCpu->CTX_SUFF(pVM);
7797 if ( !pVCpu->iem.s.aMemBbMappings[iMemMap].fUnassigned
7798 && !IEM_VERIFICATION_ENABLED(pVCpu))
7799 {
7800 uint16_t const cbFirst = pVCpu->iem.s.aMemBbMappings[iMemMap].cbFirst;
7801 uint16_t const cbSecond = pVCpu->iem.s.aMemBbMappings[iMemMap].cbSecond;
7802 uint8_t const *pbBuf = &pVCpu->iem.s.aBounceBuffers[iMemMap].ab[0];
7803 if (!pVCpu->iem.s.fBypassHandlers)
7804 {
7805 /*
7806 * Carefully and efficiently dealing with access handler return
7807 * codes make this a little bloated.
7808 */
7809 VBOXSTRICTRC rcStrict = PGMPhysWrite(pVM,
7810 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysFirst,
7811 pbBuf,
7812 cbFirst,
7813 PGMACCESSORIGIN_IEM);
7814 if (rcStrict == VINF_SUCCESS)
7815 {
7816 if (cbSecond)
7817 {
7818 rcStrict = PGMPhysWrite(pVM,
7819 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysSecond,
7820 pbBuf + cbFirst,
7821 cbSecond,
7822 PGMACCESSORIGIN_IEM);
7823 if (rcStrict == VINF_SUCCESS)
7824 { /* nothing */ }
7825 else if (PGM_PHYS_RW_IS_SUCCESS(rcStrict))
7826 {
7827 Log(("iemMemBounceBufferCommitAndUnmap: PGMPhysWrite GCPhysFirst=%RGp/%#x GCPhysSecond=%RGp/%#x %Rrc\n",
7828 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysFirst, cbFirst,
7829 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysSecond, cbSecond, VBOXSTRICTRC_VAL(rcStrict) ));
7830 rcStrict = iemSetPassUpStatus(pVCpu, rcStrict);
7831 }
7832# ifndef IN_RING3
7833 else if (fPostponeFail)
7834 {
7835 Log(("iemMemBounceBufferCommitAndUnmap: PGMPhysWrite GCPhysFirst=%RGp/%#x GCPhysSecond=%RGp/%#x %Rrc (postponed)\n",
7836 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysFirst, cbFirst,
7837 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysSecond, cbSecond, VBOXSTRICTRC_VAL(rcStrict) ));
7838 pVCpu->iem.s.aMemMappings[iMemMap].fAccess |= IEM_ACCESS_PENDING_R3_WRITE_2ND;
7839 VMCPU_FF_SET(pVCpu, VMCPU_FF_IEM);
7840 return iemSetPassUpStatus(pVCpu, rcStrict);
7841 }
7842# endif
7843 else
7844 {
7845 Log(("iemMemBounceBufferCommitAndUnmap: PGMPhysWrite GCPhysFirst=%RGp/%#x GCPhysSecond=%RGp/%#x %Rrc (!!)\n",
7846 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysFirst, cbFirst,
7847 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysSecond, cbSecond, VBOXSTRICTRC_VAL(rcStrict) ));
7848 return rcStrict;
7849 }
7850 }
7851 }
7852 else if (PGM_PHYS_RW_IS_SUCCESS(rcStrict))
7853 {
7854 if (!cbSecond)
7855 {
7856 Log(("iemMemBounceBufferCommitAndUnmap: PGMPhysWrite GCPhysFirst=%RGp/%#x %Rrc\n",
7857 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysFirst, cbFirst, VBOXSTRICTRC_VAL(rcStrict) ));
7858 rcStrict = iemSetPassUpStatus(pVCpu, rcStrict);
7859 }
7860 else
7861 {
7862 VBOXSTRICTRC rcStrict2 = PGMPhysWrite(pVM,
7863 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysSecond,
7864 pbBuf + cbFirst,
7865 cbSecond,
7866 PGMACCESSORIGIN_IEM);
7867 if (rcStrict2 == VINF_SUCCESS)
7868 {
7869 Log(("iemMemBounceBufferCommitAndUnmap: PGMPhysWrite GCPhysFirst=%RGp/%#x %Rrc GCPhysSecond=%RGp/%#x\n",
7870 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysFirst, cbFirst, VBOXSTRICTRC_VAL(rcStrict),
7871 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysSecond, cbSecond));
7872 rcStrict = iemSetPassUpStatus(pVCpu, rcStrict);
7873 }
7874 else if (PGM_PHYS_RW_IS_SUCCESS(rcStrict2))
7875 {
7876 Log(("iemMemBounceBufferCommitAndUnmap: PGMPhysWrite GCPhysFirst=%RGp/%#x %Rrc GCPhysSecond=%RGp/%#x %Rrc\n",
7877 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysFirst, cbFirst, VBOXSTRICTRC_VAL(rcStrict),
7878 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysSecond, cbSecond, VBOXSTRICTRC_VAL(rcStrict2) ));
7879 PGM_PHYS_RW_DO_UPDATE_STRICT_RC(rcStrict, rcStrict2);
7880 rcStrict = iemSetPassUpStatus(pVCpu, rcStrict);
7881 }
7882# ifndef IN_RING3
7883 else if (fPostponeFail)
7884 {
7885 Log(("iemMemBounceBufferCommitAndUnmap: PGMPhysWrite GCPhysFirst=%RGp/%#x GCPhysSecond=%RGp/%#x %Rrc (postponed)\n",
7886 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysFirst, cbFirst,
7887 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysSecond, cbSecond, VBOXSTRICTRC_VAL(rcStrict) ));
7888 pVCpu->iem.s.aMemMappings[iMemMap].fAccess |= IEM_ACCESS_PENDING_R3_WRITE_2ND;
7889 VMCPU_FF_SET(pVCpu, VMCPU_FF_IEM);
7890 return iemSetPassUpStatus(pVCpu, rcStrict);
7891 }
7892# endif
7893 else
7894 {
7895 Log(("iemMemBounceBufferCommitAndUnmap: PGMPhysWrite GCPhysFirst=%RGp/%#x %Rrc GCPhysSecond=%RGp/%#x %Rrc (!!)\n",
7896 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysFirst, cbFirst, VBOXSTRICTRC_VAL(rcStrict),
7897 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysSecond, cbSecond, VBOXSTRICTRC_VAL(rcStrict2) ));
7898 return rcStrict2;
7899 }
7900 }
7901 }
7902# ifndef IN_RING3
7903 else if (fPostponeFail)
7904 {
7905 Log(("iemMemBounceBufferCommitAndUnmap: PGMPhysWrite GCPhysFirst=%RGp/%#x GCPhysSecond=%RGp/%#x %Rrc (postponed)\n",
7906 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysFirst, cbFirst,
7907 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysSecond, cbSecond, VBOXSTRICTRC_VAL(rcStrict) ));
7908 if (!cbSecond)
7909 pVCpu->iem.s.aMemMappings[iMemMap].fAccess |= IEM_ACCESS_PENDING_R3_WRITE_1ST;
7910 else
7911 pVCpu->iem.s.aMemMappings[iMemMap].fAccess |= IEM_ACCESS_PENDING_R3_WRITE_1ST | IEM_ACCESS_PENDING_R3_WRITE_2ND;
7912 VMCPU_FF_SET(pVCpu, VMCPU_FF_IEM);
7913 return iemSetPassUpStatus(pVCpu, rcStrict);
7914 }
7915# endif
7916 else
7917 {
7918 Log(("iemMemBounceBufferCommitAndUnmap: PGMPhysWrite GCPhysFirst=%RGp/%#x %Rrc [GCPhysSecond=%RGp/%#x] (!!)\n",
7919 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysFirst, cbFirst, VBOXSTRICTRC_VAL(rcStrict),
7920 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysSecond, cbSecond));
7921 return rcStrict;
7922 }
7923 }
7924 else
7925 {
7926 /*
7927 * No access handlers, much simpler.
7928 */
7929 int rc = PGMPhysSimpleWriteGCPhys(pVM, pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysFirst, pbBuf, cbFirst);
7930 if (RT_SUCCESS(rc))
7931 {
7932 if (cbSecond)
7933 {
7934 rc = PGMPhysSimpleWriteGCPhys(pVM, pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysSecond, pbBuf + cbFirst, cbSecond);
7935 if (RT_SUCCESS(rc))
7936 { /* likely */ }
7937 else
7938 {
7939 Log(("iemMemBounceBufferCommitAndUnmap: PGMPhysSimpleWriteGCPhys GCPhysFirst=%RGp/%#x GCPhysSecond=%RGp/%#x %Rrc (!!)\n",
7940 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysFirst, cbFirst,
7941 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysSecond, cbSecond, rc));
7942 return rc;
7943 }
7944 }
7945 }
7946 else
7947 {
7948 Log(("iemMemBounceBufferCommitAndUnmap: PGMPhysSimpleWriteGCPhys GCPhysFirst=%RGp/%#x %Rrc [GCPhysSecond=%RGp/%#x] (!!)\n",
7949 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysFirst, cbFirst, rc,
7950 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysSecond, cbSecond));
7951 return rc;
7952 }
7953 }
7954 }
7955#endif
7956
7957#if defined(IEM_VERIFICATION_MODE_FULL) && defined(IN_RING3)
7958 /*
7959 * Record the write(s).
7960 */
7961 if (!pVCpu->iem.s.fNoRem)
7962 {
7963 PIEMVERIFYEVTREC pEvtRec = iemVerifyAllocRecord(pVCpu);
7964 if (pEvtRec)
7965 {
7966 pEvtRec->enmEvent = IEMVERIFYEVENT_RAM_WRITE;
7967 pEvtRec->u.RamWrite.GCPhys = pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysFirst;
7968 pEvtRec->u.RamWrite.cb = pVCpu->iem.s.aMemBbMappings[iMemMap].cbFirst;
7969 memcpy(pEvtRec->u.RamWrite.ab, &pVCpu->iem.s.aBounceBuffers[iMemMap].ab[0], pVCpu->iem.s.aMemBbMappings[iMemMap].cbFirst);
7970 AssertCompile(sizeof(pEvtRec->u.RamWrite.ab) == sizeof(pVCpu->iem.s.aBounceBuffers[0].ab));
7971 pEvtRec->pNext = *pVCpu->iem.s.ppIemEvtRecNext;
7972 *pVCpu->iem.s.ppIemEvtRecNext = pEvtRec;
7973 }
7974 if (pVCpu->iem.s.aMemBbMappings[iMemMap].cbSecond)
7975 {
7976 pEvtRec = iemVerifyAllocRecord(pVCpu);
7977 if (pEvtRec)
7978 {
7979 pEvtRec->enmEvent = IEMVERIFYEVENT_RAM_WRITE;
7980 pEvtRec->u.RamWrite.GCPhys = pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysSecond;
7981 pEvtRec->u.RamWrite.cb = pVCpu->iem.s.aMemBbMappings[iMemMap].cbSecond;
7982 memcpy(pEvtRec->u.RamWrite.ab,
7983 &pVCpu->iem.s.aBounceBuffers[iMemMap].ab[pVCpu->iem.s.aMemBbMappings[iMemMap].cbFirst],
7984 pVCpu->iem.s.aMemBbMappings[iMemMap].cbSecond);
7985 pEvtRec->pNext = *pVCpu->iem.s.ppIemEvtRecNext;
7986 *pVCpu->iem.s.ppIemEvtRecNext = pEvtRec;
7987 }
7988 }
7989 }
7990#endif
7991#if defined(IEM_VERIFICATION_MODE_MINIMAL) || defined(IEM_LOG_MEMORY_WRITES)
7992 Log(("IEM Wrote %RGp: %.*Rhxs\n", pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysFirst,
7993 RT_MAX(RT_MIN(pVCpu->iem.s.aMemBbMappings[iMemMap].cbFirst, 64), 1), &pVCpu->iem.s.aBounceBuffers[iMemMap].ab[0]));
7994 if (pVCpu->iem.s.aMemBbMappings[iMemMap].cbSecond)
7995 Log(("IEM Wrote %RGp: %.*Rhxs [2nd page]\n", pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysSecond,
7996 RT_MIN(pVCpu->iem.s.aMemBbMappings[iMemMap].cbSecond, 64),
7997 &pVCpu->iem.s.aBounceBuffers[iMemMap].ab[pVCpu->iem.s.aMemBbMappings[iMemMap].cbFirst]));
7998
7999 size_t cbWrote = pVCpu->iem.s.aMemBbMappings[iMemMap].cbFirst + pVCpu->iem.s.aMemBbMappings[iMemMap].cbSecond;
8000 g_cbIemWrote = cbWrote;
8001 memcpy(g_abIemWrote, &pVCpu->iem.s.aBounceBuffers[iMemMap].ab[0], RT_MIN(cbWrote, sizeof(g_abIemWrote)));
8002#endif
8003
8004 /*
8005 * Free the mapping entry.
8006 */
8007 pVCpu->iem.s.aMemMappings[iMemMap].fAccess = IEM_ACCESS_INVALID;
8008 Assert(pVCpu->iem.s.cActiveMappings != 0);
8009 pVCpu->iem.s.cActiveMappings--;
8010 return VINF_SUCCESS;
8011}
8012
8013
8014/**
8015 * iemMemMap worker that deals with a request crossing pages.
8016 */
8017IEM_STATIC VBOXSTRICTRC
8018iemMemBounceBufferMapCrossPage(PVMCPU pVCpu, int iMemMap, void **ppvMem, size_t cbMem, RTGCPTR GCPtrFirst, uint32_t fAccess)
8019{
8020 /*
8021 * Do the address translations.
8022 */
8023 RTGCPHYS GCPhysFirst;
8024 VBOXSTRICTRC rcStrict = iemMemPageTranslateAndCheckAccess(pVCpu, GCPtrFirst, fAccess, &GCPhysFirst);
8025 if (rcStrict != VINF_SUCCESS)
8026 return rcStrict;
8027
8028 RTGCPHYS GCPhysSecond;
8029 rcStrict = iemMemPageTranslateAndCheckAccess(pVCpu, (GCPtrFirst + (cbMem - 1)) & ~(RTGCPTR)PAGE_OFFSET_MASK,
8030 fAccess, &GCPhysSecond);
8031 if (rcStrict != VINF_SUCCESS)
8032 return rcStrict;
8033 GCPhysSecond &= ~(RTGCPHYS)PAGE_OFFSET_MASK;
8034
8035 PVM pVM = pVCpu->CTX_SUFF(pVM);
8036#ifdef IEM_VERIFICATION_MODE_FULL
8037 /*
8038 * Detect problematic memory when verifying so we can select
8039 * the right execution engine. (TLB: Redo this.)
8040 */
8041 if (IEM_FULL_VERIFICATION_ENABLED(pVCpu))
8042 {
8043 int rc2 = PGMPhysIemQueryAccess(pVM, GCPhysFirst, RT_BOOL(fAccess & IEM_ACCESS_TYPE_WRITE), pVCpu->iem.s.fBypassHandlers);
8044 if (RT_SUCCESS(rc2))
8045 rc2 = PGMPhysIemQueryAccess(pVM, GCPhysSecond, RT_BOOL(fAccess & IEM_ACCESS_TYPE_WRITE), pVCpu->iem.s.fBypassHandlers);
8046 if (RT_FAILURE(rc2))
8047 pVCpu->iem.s.fProblematicMemory = true;
8048 }
8049#endif
8050
8051
8052 /*
8053 * Read in the current memory content if it's a read, execute or partial
8054 * write access.
8055 */
8056 uint8_t *pbBuf = &pVCpu->iem.s.aBounceBuffers[iMemMap].ab[0];
8057 uint32_t const cbFirstPage = PAGE_SIZE - (GCPhysFirst & PAGE_OFFSET_MASK);
8058 uint32_t const cbSecondPage = (uint32_t)(cbMem - cbFirstPage);
8059
8060 if (fAccess & (IEM_ACCESS_TYPE_READ | IEM_ACCESS_TYPE_EXEC | IEM_ACCESS_PARTIAL_WRITE))
8061 {
8062 if (!pVCpu->iem.s.fBypassHandlers)
8063 {
8064 /*
8065 * Must carefully deal with access handler status codes here,
8066 * makes the code a bit bloated.
8067 */
8068 rcStrict = PGMPhysRead(pVM, GCPhysFirst, pbBuf, cbFirstPage, PGMACCESSORIGIN_IEM);
8069 if (rcStrict == VINF_SUCCESS)
8070 {
8071 rcStrict = PGMPhysRead(pVM, GCPhysSecond, pbBuf + cbFirstPage, cbSecondPage, PGMACCESSORIGIN_IEM);
8072 if (rcStrict == VINF_SUCCESS)
8073 { /*likely */ }
8074 else if (PGM_PHYS_RW_IS_SUCCESS(rcStrict))
8075 rcStrict = iemSetPassUpStatus(pVCpu, rcStrict);
8076 else
8077 {
8078 Log(("iemMemBounceBufferMapPhys: PGMPhysRead GCPhysSecond=%RGp rcStrict2=%Rrc (!!)\n",
8079 GCPhysSecond, VBOXSTRICTRC_VAL(rcStrict) ));
8080 return rcStrict;
8081 }
8082 }
8083 else if (PGM_PHYS_RW_IS_SUCCESS(rcStrict))
8084 {
8085 VBOXSTRICTRC rcStrict2 = PGMPhysRead(pVM, GCPhysSecond, pbBuf + cbFirstPage, cbSecondPage, PGMACCESSORIGIN_IEM);
8086 if (PGM_PHYS_RW_IS_SUCCESS(rcStrict2))
8087 {
8088 PGM_PHYS_RW_DO_UPDATE_STRICT_RC(rcStrict, rcStrict2);
8089 rcStrict = iemSetPassUpStatus(pVCpu, rcStrict);
8090 }
8091 else
8092 {
8093 Log(("iemMemBounceBufferMapPhys: PGMPhysRead GCPhysSecond=%RGp rcStrict2=%Rrc (rcStrict=%Rrc) (!!)\n",
8094 GCPhysSecond, VBOXSTRICTRC_VAL(rcStrict2), VBOXSTRICTRC_VAL(rcStrict2) ));
8095 return rcStrict2;
8096 }
8097 }
8098 else
8099 {
8100 Log(("iemMemBounceBufferMapPhys: PGMPhysRead GCPhysFirst=%RGp rcStrict=%Rrc (!!)\n",
8101 GCPhysFirst, VBOXSTRICTRC_VAL(rcStrict) ));
8102 return rcStrict;
8103 }
8104 }
8105 else
8106 {
8107 /*
8108 * No informational status codes here, much more straight forward.
8109 */
8110 int rc = PGMPhysSimpleReadGCPhys(pVM, pbBuf, GCPhysFirst, cbFirstPage);
8111 if (RT_SUCCESS(rc))
8112 {
8113 Assert(rc == VINF_SUCCESS);
8114 rc = PGMPhysSimpleReadGCPhys(pVM, pbBuf + cbFirstPage, GCPhysSecond, cbSecondPage);
8115 if (RT_SUCCESS(rc))
8116 Assert(rc == VINF_SUCCESS);
8117 else
8118 {
8119 Log(("iemMemBounceBufferMapPhys: PGMPhysSimpleReadGCPhys GCPhysSecond=%RGp rc=%Rrc (!!)\n", GCPhysSecond, rc));
8120 return rc;
8121 }
8122 }
8123 else
8124 {
8125 Log(("iemMemBounceBufferMapPhys: PGMPhysSimpleReadGCPhys GCPhysFirst=%RGp rc=%Rrc (!!)\n", GCPhysFirst, rc));
8126 return rc;
8127 }
8128 }
8129
8130#if defined(IEM_VERIFICATION_MODE_FULL) && defined(IN_RING3)
8131 if ( !pVCpu->iem.s.fNoRem
8132 && (fAccess & (IEM_ACCESS_TYPE_READ | IEM_ACCESS_TYPE_EXEC)) )
8133 {
8134 /*
8135 * Record the reads.
8136 */
8137 PIEMVERIFYEVTREC pEvtRec = iemVerifyAllocRecord(pVCpu);
8138 if (pEvtRec)
8139 {
8140 pEvtRec->enmEvent = IEMVERIFYEVENT_RAM_READ;
8141 pEvtRec->u.RamRead.GCPhys = GCPhysFirst;
8142 pEvtRec->u.RamRead.cb = cbFirstPage;
8143 pEvtRec->pNext = *pVCpu->iem.s.ppIemEvtRecNext;
8144 *pVCpu->iem.s.ppIemEvtRecNext = pEvtRec;
8145 }
8146 pEvtRec = iemVerifyAllocRecord(pVCpu);
8147 if (pEvtRec)
8148 {
8149 pEvtRec->enmEvent = IEMVERIFYEVENT_RAM_READ;
8150 pEvtRec->u.RamRead.GCPhys = GCPhysSecond;
8151 pEvtRec->u.RamRead.cb = cbSecondPage;
8152 pEvtRec->pNext = *pVCpu->iem.s.ppIemEvtRecNext;
8153 *pVCpu->iem.s.ppIemEvtRecNext = pEvtRec;
8154 }
8155 }
8156#endif
8157 }
8158#ifdef VBOX_STRICT
8159 else
8160 memset(pbBuf, 0xcc, cbMem);
8161 if (cbMem < sizeof(pVCpu->iem.s.aBounceBuffers[iMemMap].ab))
8162 memset(pbBuf + cbMem, 0xaa, sizeof(pVCpu->iem.s.aBounceBuffers[iMemMap].ab) - cbMem);
8163#endif
8164
8165 /*
8166 * Commit the bounce buffer entry.
8167 */
8168 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysFirst = GCPhysFirst;
8169 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysSecond = GCPhysSecond;
8170 pVCpu->iem.s.aMemBbMappings[iMemMap].cbFirst = (uint16_t)cbFirstPage;
8171 pVCpu->iem.s.aMemBbMappings[iMemMap].cbSecond = (uint16_t)cbSecondPage;
8172 pVCpu->iem.s.aMemBbMappings[iMemMap].fUnassigned = false;
8173 pVCpu->iem.s.aMemMappings[iMemMap].pv = pbBuf;
8174 pVCpu->iem.s.aMemMappings[iMemMap].fAccess = fAccess | IEM_ACCESS_BOUNCE_BUFFERED;
8175 pVCpu->iem.s.iNextMapping = iMemMap + 1;
8176 pVCpu->iem.s.cActiveMappings++;
8177
8178 iemMemUpdateWrittenCounter(pVCpu, fAccess, cbMem);
8179 *ppvMem = pbBuf;
8180 return VINF_SUCCESS;
8181}
8182
8183
8184/**
8185 * iemMemMap woker that deals with iemMemPageMap failures.
8186 */
8187IEM_STATIC VBOXSTRICTRC iemMemBounceBufferMapPhys(PVMCPU pVCpu, unsigned iMemMap, void **ppvMem, size_t cbMem,
8188 RTGCPHYS GCPhysFirst, uint32_t fAccess, VBOXSTRICTRC rcMap)
8189{
8190 /*
8191 * Filter out conditions we can handle and the ones which shouldn't happen.
8192 */
8193 if ( rcMap != VERR_PGM_PHYS_TLB_CATCH_WRITE
8194 && rcMap != VERR_PGM_PHYS_TLB_CATCH_ALL
8195 && rcMap != VERR_PGM_PHYS_TLB_UNASSIGNED)
8196 {
8197 AssertReturn(RT_FAILURE_NP(rcMap), VERR_IEM_IPE_8);
8198 return rcMap;
8199 }
8200 pVCpu->iem.s.cPotentialExits++;
8201
8202 /*
8203 * Read in the current memory content if it's a read, execute or partial
8204 * write access.
8205 */
8206 uint8_t *pbBuf = &pVCpu->iem.s.aBounceBuffers[iMemMap].ab[0];
8207 if (fAccess & (IEM_ACCESS_TYPE_READ | IEM_ACCESS_TYPE_EXEC | IEM_ACCESS_PARTIAL_WRITE))
8208 {
8209 if (rcMap == VERR_PGM_PHYS_TLB_UNASSIGNED)
8210 memset(pbBuf, 0xff, cbMem);
8211 else
8212 {
8213 int rc;
8214 if (!pVCpu->iem.s.fBypassHandlers)
8215 {
8216 VBOXSTRICTRC rcStrict = PGMPhysRead(pVCpu->CTX_SUFF(pVM), GCPhysFirst, pbBuf, cbMem, PGMACCESSORIGIN_IEM);
8217 if (rcStrict == VINF_SUCCESS)
8218 { /* nothing */ }
8219 else if (PGM_PHYS_RW_IS_SUCCESS(rcStrict))
8220 rcStrict = iemSetPassUpStatus(pVCpu, rcStrict);
8221 else
8222 {
8223 Log(("iemMemBounceBufferMapPhys: PGMPhysRead GCPhysFirst=%RGp rcStrict=%Rrc (!!)\n",
8224 GCPhysFirst, VBOXSTRICTRC_VAL(rcStrict) ));
8225 return rcStrict;
8226 }
8227 }
8228 else
8229 {
8230 rc = PGMPhysSimpleReadGCPhys(pVCpu->CTX_SUFF(pVM), pbBuf, GCPhysFirst, cbMem);
8231 if (RT_SUCCESS(rc))
8232 { /* likely */ }
8233 else
8234 {
8235 Log(("iemMemBounceBufferMapPhys: PGMPhysSimpleReadGCPhys GCPhysFirst=%RGp rcStrict=%Rrc (!!)\n",
8236 GCPhysFirst, rc));
8237 return rc;
8238 }
8239 }
8240 }
8241
8242#if defined(IEM_VERIFICATION_MODE_FULL) && defined(IN_RING3)
8243 if ( !pVCpu->iem.s.fNoRem
8244 && (fAccess & (IEM_ACCESS_TYPE_READ | IEM_ACCESS_TYPE_EXEC)) )
8245 {
8246 /*
8247 * Record the read.
8248 */
8249 PIEMVERIFYEVTREC pEvtRec = iemVerifyAllocRecord(pVCpu);
8250 if (pEvtRec)
8251 {
8252 pEvtRec->enmEvent = IEMVERIFYEVENT_RAM_READ;
8253 pEvtRec->u.RamRead.GCPhys = GCPhysFirst;
8254 pEvtRec->u.RamRead.cb = (uint32_t)cbMem;
8255 pEvtRec->pNext = *pVCpu->iem.s.ppIemEvtRecNext;
8256 *pVCpu->iem.s.ppIemEvtRecNext = pEvtRec;
8257 }
8258 }
8259#endif
8260 }
8261#ifdef VBOX_STRICT
8262 else
8263 memset(pbBuf, 0xcc, cbMem);
8264#endif
8265#ifdef VBOX_STRICT
8266 if (cbMem < sizeof(pVCpu->iem.s.aBounceBuffers[iMemMap].ab))
8267 memset(pbBuf + cbMem, 0xaa, sizeof(pVCpu->iem.s.aBounceBuffers[iMemMap].ab) - cbMem);
8268#endif
8269
8270 /*
8271 * Commit the bounce buffer entry.
8272 */
8273 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysFirst = GCPhysFirst;
8274 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysSecond = NIL_RTGCPHYS;
8275 pVCpu->iem.s.aMemBbMappings[iMemMap].cbFirst = (uint16_t)cbMem;
8276 pVCpu->iem.s.aMemBbMappings[iMemMap].cbSecond = 0;
8277 pVCpu->iem.s.aMemBbMappings[iMemMap].fUnassigned = rcMap == VERR_PGM_PHYS_TLB_UNASSIGNED;
8278 pVCpu->iem.s.aMemMappings[iMemMap].pv = pbBuf;
8279 pVCpu->iem.s.aMemMappings[iMemMap].fAccess = fAccess | IEM_ACCESS_BOUNCE_BUFFERED;
8280 pVCpu->iem.s.iNextMapping = iMemMap + 1;
8281 pVCpu->iem.s.cActiveMappings++;
8282
8283 iemMemUpdateWrittenCounter(pVCpu, fAccess, cbMem);
8284 *ppvMem = pbBuf;
8285 return VINF_SUCCESS;
8286}
8287
8288
8289
8290/**
8291 * Maps the specified guest memory for the given kind of access.
8292 *
8293 * This may be using bounce buffering of the memory if it's crossing a page
8294 * boundary or if there is an access handler installed for any of it. Because
8295 * of lock prefix guarantees, we're in for some extra clutter when this
8296 * happens.
8297 *
8298 * This may raise a \#GP, \#SS, \#PF or \#AC.
8299 *
8300 * @returns VBox strict status code.
8301 *
8302 * @param pVCpu The cross context virtual CPU structure of the calling thread.
8303 * @param ppvMem Where to return the pointer to the mapped
8304 * memory.
8305 * @param cbMem The number of bytes to map. This is usually 1,
8306 * 2, 4, 6, 8, 12, 16, 32 or 512. When used by
8307 * string operations it can be up to a page.
8308 * @param iSegReg The index of the segment register to use for
8309 * this access. The base and limits are checked.
8310 * Use UINT8_MAX to indicate that no segmentation
8311 * is required (for IDT, GDT and LDT accesses).
8312 * @param GCPtrMem The address of the guest memory.
8313 * @param fAccess How the memory is being accessed. The
8314 * IEM_ACCESS_TYPE_XXX bit is used to figure out
8315 * how to map the memory, while the
8316 * IEM_ACCESS_WHAT_XXX bit is used when raising
8317 * exceptions.
8318 */
8319IEM_STATIC VBOXSTRICTRC
8320iemMemMap(PVMCPU pVCpu, void **ppvMem, size_t cbMem, uint8_t iSegReg, RTGCPTR GCPtrMem, uint32_t fAccess)
8321{
8322 /*
8323 * Check the input and figure out which mapping entry to use.
8324 */
8325 Assert(cbMem <= 64 || cbMem == 512 || cbMem == 108 || cbMem == 104 || cbMem == 94); /* 512 is the max! */
8326 Assert(~(fAccess & ~(IEM_ACCESS_TYPE_MASK | IEM_ACCESS_WHAT_MASK)));
8327 Assert(pVCpu->iem.s.cActiveMappings < RT_ELEMENTS(pVCpu->iem.s.aMemMappings));
8328
8329 unsigned iMemMap = pVCpu->iem.s.iNextMapping;
8330 if ( iMemMap >= RT_ELEMENTS(pVCpu->iem.s.aMemMappings)
8331 || pVCpu->iem.s.aMemMappings[iMemMap].fAccess != IEM_ACCESS_INVALID)
8332 {
8333 iMemMap = iemMemMapFindFree(pVCpu);
8334 AssertLogRelMsgReturn(iMemMap < RT_ELEMENTS(pVCpu->iem.s.aMemMappings),
8335 ("active=%d fAccess[0] = {%#x, %#x, %#x}\n", pVCpu->iem.s.cActiveMappings,
8336 pVCpu->iem.s.aMemMappings[0].fAccess, pVCpu->iem.s.aMemMappings[1].fAccess,
8337 pVCpu->iem.s.aMemMappings[2].fAccess),
8338 VERR_IEM_IPE_9);
8339 }
8340
8341 /*
8342 * Map the memory, checking that we can actually access it. If something
8343 * slightly complicated happens, fall back on bounce buffering.
8344 */
8345 VBOXSTRICTRC rcStrict = iemMemApplySegment(pVCpu, fAccess, iSegReg, cbMem, &GCPtrMem);
8346 if (rcStrict != VINF_SUCCESS)
8347 return rcStrict;
8348
8349 if ((GCPtrMem & PAGE_OFFSET_MASK) + cbMem > PAGE_SIZE) /* Crossing a page boundary? */
8350 return iemMemBounceBufferMapCrossPage(pVCpu, iMemMap, ppvMem, cbMem, GCPtrMem, fAccess);
8351
8352 RTGCPHYS GCPhysFirst;
8353 rcStrict = iemMemPageTranslateAndCheckAccess(pVCpu, GCPtrMem, fAccess, &GCPhysFirst);
8354 if (rcStrict != VINF_SUCCESS)
8355 return rcStrict;
8356
8357 if (fAccess & IEM_ACCESS_TYPE_WRITE)
8358 Log8(("IEM WR %RGv (%RGp) LB %#zx\n", GCPtrMem, GCPhysFirst, cbMem));
8359 if (fAccess & IEM_ACCESS_TYPE_READ)
8360 Log9(("IEM RD %RGv (%RGp) LB %#zx\n", GCPtrMem, GCPhysFirst, cbMem));
8361
8362 void *pvMem;
8363 rcStrict = iemMemPageMap(pVCpu, GCPhysFirst, fAccess, &pvMem, &pVCpu->iem.s.aMemMappingLocks[iMemMap].Lock);
8364 if (rcStrict != VINF_SUCCESS)
8365 return iemMemBounceBufferMapPhys(pVCpu, iMemMap, ppvMem, cbMem, GCPhysFirst, fAccess, rcStrict);
8366
8367 /*
8368 * Fill in the mapping table entry.
8369 */
8370 pVCpu->iem.s.aMemMappings[iMemMap].pv = pvMem;
8371 pVCpu->iem.s.aMemMappings[iMemMap].fAccess = fAccess;
8372 pVCpu->iem.s.iNextMapping = iMemMap + 1;
8373 pVCpu->iem.s.cActiveMappings++;
8374
8375 iemMemUpdateWrittenCounter(pVCpu, fAccess, cbMem);
8376 *ppvMem = pvMem;
8377 return VINF_SUCCESS;
8378}
8379
8380
8381/**
8382 * Commits the guest memory if bounce buffered and unmaps it.
8383 *
8384 * @returns Strict VBox status code.
8385 * @param pVCpu The cross context virtual CPU structure of the calling thread.
8386 * @param pvMem The mapping.
8387 * @param fAccess The kind of access.
8388 */
8389IEM_STATIC VBOXSTRICTRC iemMemCommitAndUnmap(PVMCPU pVCpu, void *pvMem, uint32_t fAccess)
8390{
8391 int iMemMap = iemMapLookup(pVCpu, pvMem, fAccess);
8392 AssertReturn(iMemMap >= 0, iMemMap);
8393
8394 /* If it's bounce buffered, we may need to write back the buffer. */
8395 if (pVCpu->iem.s.aMemMappings[iMemMap].fAccess & IEM_ACCESS_BOUNCE_BUFFERED)
8396 {
8397 if (pVCpu->iem.s.aMemMappings[iMemMap].fAccess & IEM_ACCESS_TYPE_WRITE)
8398 return iemMemBounceBufferCommitAndUnmap(pVCpu, iMemMap, false /*fPostponeFail*/);
8399 }
8400 /* Otherwise unlock it. */
8401 else
8402 PGMPhysReleasePageMappingLock(pVCpu->CTX_SUFF(pVM), &pVCpu->iem.s.aMemMappingLocks[iMemMap].Lock);
8403
8404 /* Free the entry. */
8405 pVCpu->iem.s.aMemMappings[iMemMap].fAccess = IEM_ACCESS_INVALID;
8406 Assert(pVCpu->iem.s.cActiveMappings != 0);
8407 pVCpu->iem.s.cActiveMappings--;
8408 return VINF_SUCCESS;
8409}
8410
8411#ifdef IEM_WITH_SETJMP
8412
8413/**
8414 * Maps the specified guest memory for the given kind of access, longjmp on
8415 * error.
8416 *
8417 * This may be using bounce buffering of the memory if it's crossing a page
8418 * boundary or if there is an access handler installed for any of it. Because
8419 * of lock prefix guarantees, we're in for some extra clutter when this
8420 * happens.
8421 *
8422 * This may raise a \#GP, \#SS, \#PF or \#AC.
8423 *
8424 * @returns Pointer to the mapped memory.
8425 *
8426 * @param pVCpu The cross context virtual CPU structure of the calling thread.
8427 * @param cbMem The number of bytes to map. This is usually 1,
8428 * 2, 4, 6, 8, 12, 16, 32 or 512. When used by
8429 * string operations it can be up to a page.
8430 * @param iSegReg The index of the segment register to use for
8431 * this access. The base and limits are checked.
8432 * Use UINT8_MAX to indicate that no segmentation
8433 * is required (for IDT, GDT and LDT accesses).
8434 * @param GCPtrMem The address of the guest memory.
8435 * @param fAccess How the memory is being accessed. The
8436 * IEM_ACCESS_TYPE_XXX bit is used to figure out
8437 * how to map the memory, while the
8438 * IEM_ACCESS_WHAT_XXX bit is used when raising
8439 * exceptions.
8440 */
8441IEM_STATIC void *iemMemMapJmp(PVMCPU pVCpu, size_t cbMem, uint8_t iSegReg, RTGCPTR GCPtrMem, uint32_t fAccess)
8442{
8443 /*
8444 * Check the input and figure out which mapping entry to use.
8445 */
8446 Assert(cbMem <= 64 || cbMem == 512 || cbMem == 108 || cbMem == 104 || cbMem == 94); /* 512 is the max! */
8447 Assert(~(fAccess & ~(IEM_ACCESS_TYPE_MASK | IEM_ACCESS_WHAT_MASK)));
8448 Assert(pVCpu->iem.s.cActiveMappings < RT_ELEMENTS(pVCpu->iem.s.aMemMappings));
8449
8450 unsigned iMemMap = pVCpu->iem.s.iNextMapping;
8451 if ( iMemMap >= RT_ELEMENTS(pVCpu->iem.s.aMemMappings)
8452 || pVCpu->iem.s.aMemMappings[iMemMap].fAccess != IEM_ACCESS_INVALID)
8453 {
8454 iMemMap = iemMemMapFindFree(pVCpu);
8455 AssertLogRelMsgStmt(iMemMap < RT_ELEMENTS(pVCpu->iem.s.aMemMappings),
8456 ("active=%d fAccess[0] = {%#x, %#x, %#x}\n", pVCpu->iem.s.cActiveMappings,
8457 pVCpu->iem.s.aMemMappings[0].fAccess, pVCpu->iem.s.aMemMappings[1].fAccess,
8458 pVCpu->iem.s.aMemMappings[2].fAccess),
8459 longjmp(*pVCpu->iem.s.CTX_SUFF(pJmpBuf), VERR_IEM_IPE_9));
8460 }
8461
8462 /*
8463 * Map the memory, checking that we can actually access it. If something
8464 * slightly complicated happens, fall back on bounce buffering.
8465 */
8466 VBOXSTRICTRC rcStrict = iemMemApplySegment(pVCpu, fAccess, iSegReg, cbMem, &GCPtrMem);
8467 if (rcStrict == VINF_SUCCESS) { /*likely*/ }
8468 else longjmp(*pVCpu->iem.s.CTX_SUFF(pJmpBuf), VBOXSTRICTRC_VAL(rcStrict));
8469
8470 /* Crossing a page boundary? */
8471 if ((GCPtrMem & PAGE_OFFSET_MASK) + cbMem <= PAGE_SIZE)
8472 { /* No (likely). */ }
8473 else
8474 {
8475 void *pvMem;
8476 rcStrict = iemMemBounceBufferMapCrossPage(pVCpu, iMemMap, &pvMem, cbMem, GCPtrMem, fAccess);
8477 if (rcStrict == VINF_SUCCESS)
8478 return pvMem;
8479 longjmp(*pVCpu->iem.s.CTX_SUFF(pJmpBuf), VBOXSTRICTRC_VAL(rcStrict));
8480 }
8481
8482 RTGCPHYS GCPhysFirst;
8483 rcStrict = iemMemPageTranslateAndCheckAccess(pVCpu, GCPtrMem, fAccess, &GCPhysFirst);
8484 if (rcStrict == VINF_SUCCESS) { /*likely*/ }
8485 else longjmp(*pVCpu->iem.s.CTX_SUFF(pJmpBuf), VBOXSTRICTRC_VAL(rcStrict));
8486
8487 if (fAccess & IEM_ACCESS_TYPE_WRITE)
8488 Log8(("IEM WR %RGv (%RGp) LB %#zx\n", GCPtrMem, GCPhysFirst, cbMem));
8489 if (fAccess & IEM_ACCESS_TYPE_READ)
8490 Log9(("IEM RD %RGv (%RGp) LB %#zx\n", GCPtrMem, GCPhysFirst, cbMem));
8491
8492 void *pvMem;
8493 rcStrict = iemMemPageMap(pVCpu, GCPhysFirst, fAccess, &pvMem, &pVCpu->iem.s.aMemMappingLocks[iMemMap].Lock);
8494 if (rcStrict == VINF_SUCCESS)
8495 { /* likely */ }
8496 else
8497 {
8498 rcStrict = iemMemBounceBufferMapPhys(pVCpu, iMemMap, &pvMem, cbMem, GCPhysFirst, fAccess, rcStrict);
8499 if (rcStrict == VINF_SUCCESS)
8500 return pvMem;
8501 longjmp(*pVCpu->iem.s.CTX_SUFF(pJmpBuf), VBOXSTRICTRC_VAL(rcStrict));
8502 }
8503
8504 /*
8505 * Fill in the mapping table entry.
8506 */
8507 pVCpu->iem.s.aMemMappings[iMemMap].pv = pvMem;
8508 pVCpu->iem.s.aMemMappings[iMemMap].fAccess = fAccess;
8509 pVCpu->iem.s.iNextMapping = iMemMap + 1;
8510 pVCpu->iem.s.cActiveMappings++;
8511
8512 iemMemUpdateWrittenCounter(pVCpu, fAccess, cbMem);
8513 return pvMem;
8514}
8515
8516
8517/**
8518 * Commits the guest memory if bounce buffered and unmaps it, longjmp on error.
8519 *
8520 * @param pVCpu The cross context virtual CPU structure of the calling thread.
8521 * @param pvMem The mapping.
8522 * @param fAccess The kind of access.
8523 */
8524IEM_STATIC void iemMemCommitAndUnmapJmp(PVMCPU pVCpu, void *pvMem, uint32_t fAccess)
8525{
8526 int iMemMap = iemMapLookup(pVCpu, pvMem, fAccess);
8527 AssertStmt(iMemMap >= 0, longjmp(*pVCpu->iem.s.CTX_SUFF(pJmpBuf), iMemMap));
8528
8529 /* If it's bounce buffered, we may need to write back the buffer. */
8530 if (pVCpu->iem.s.aMemMappings[iMemMap].fAccess & IEM_ACCESS_BOUNCE_BUFFERED)
8531 {
8532 if (pVCpu->iem.s.aMemMappings[iMemMap].fAccess & IEM_ACCESS_TYPE_WRITE)
8533 {
8534 VBOXSTRICTRC rcStrict = iemMemBounceBufferCommitAndUnmap(pVCpu, iMemMap, false /*fPostponeFail*/);
8535 if (rcStrict == VINF_SUCCESS)
8536 return;
8537 longjmp(*pVCpu->iem.s.CTX_SUFF(pJmpBuf), VBOXSTRICTRC_VAL(rcStrict));
8538 }
8539 }
8540 /* Otherwise unlock it. */
8541 else
8542 PGMPhysReleasePageMappingLock(pVCpu->CTX_SUFF(pVM), &pVCpu->iem.s.aMemMappingLocks[iMemMap].Lock);
8543
8544 /* Free the entry. */
8545 pVCpu->iem.s.aMemMappings[iMemMap].fAccess = IEM_ACCESS_INVALID;
8546 Assert(pVCpu->iem.s.cActiveMappings != 0);
8547 pVCpu->iem.s.cActiveMappings--;
8548}
8549
8550#endif
8551
8552#ifndef IN_RING3
8553/**
8554 * Commits the guest memory if bounce buffered and unmaps it, if any bounce
8555 * buffer part shows trouble it will be postponed to ring-3 (sets FF and stuff).
8556 *
8557 * Allows the instruction to be completed and retired, while the IEM user will
8558 * return to ring-3 immediately afterwards and do the postponed writes there.
8559 *
8560 * @returns VBox status code (no strict statuses). Caller must check
8561 * VMCPU_FF_IEM before repeating string instructions and similar stuff.
8562 * @param pVCpu The cross context virtual CPU structure of the calling thread.
8563 * @param pvMem The mapping.
8564 * @param fAccess The kind of access.
8565 */
8566IEM_STATIC VBOXSTRICTRC iemMemCommitAndUnmapPostponeTroubleToR3(PVMCPU pVCpu, void *pvMem, uint32_t fAccess)
8567{
8568 int iMemMap = iemMapLookup(pVCpu, pvMem, fAccess);
8569 AssertReturn(iMemMap >= 0, iMemMap);
8570
8571 /* If it's bounce buffered, we may need to write back the buffer. */
8572 if (pVCpu->iem.s.aMemMappings[iMemMap].fAccess & IEM_ACCESS_BOUNCE_BUFFERED)
8573 {
8574 if (pVCpu->iem.s.aMemMappings[iMemMap].fAccess & IEM_ACCESS_TYPE_WRITE)
8575 return iemMemBounceBufferCommitAndUnmap(pVCpu, iMemMap, true /*fPostponeFail*/);
8576 }
8577 /* Otherwise unlock it. */
8578 else
8579 PGMPhysReleasePageMappingLock(pVCpu->CTX_SUFF(pVM), &pVCpu->iem.s.aMemMappingLocks[iMemMap].Lock);
8580
8581 /* Free the entry. */
8582 pVCpu->iem.s.aMemMappings[iMemMap].fAccess = IEM_ACCESS_INVALID;
8583 Assert(pVCpu->iem.s.cActiveMappings != 0);
8584 pVCpu->iem.s.cActiveMappings--;
8585 return VINF_SUCCESS;
8586}
8587#endif
8588
8589
8590/**
8591 * Rollbacks mappings, releasing page locks and such.
8592 *
8593 * The caller shall only call this after checking cActiveMappings.
8594 *
8595 * @returns Strict VBox status code to pass up.
8596 * @param pVCpu The cross context virtual CPU structure of the calling thread.
8597 */
8598IEM_STATIC void iemMemRollback(PVMCPU pVCpu)
8599{
8600 Assert(pVCpu->iem.s.cActiveMappings > 0);
8601
8602 uint32_t iMemMap = RT_ELEMENTS(pVCpu->iem.s.aMemMappings);
8603 while (iMemMap-- > 0)
8604 {
8605 uint32_t fAccess = pVCpu->iem.s.aMemMappings[iMemMap].fAccess;
8606 if (fAccess != IEM_ACCESS_INVALID)
8607 {
8608 AssertMsg(!(fAccess & ~IEM_ACCESS_VALID_MASK) && fAccess != 0, ("%#x\n", fAccess));
8609 pVCpu->iem.s.aMemMappings[iMemMap].fAccess = IEM_ACCESS_INVALID;
8610 if (!(fAccess & IEM_ACCESS_BOUNCE_BUFFERED))
8611 PGMPhysReleasePageMappingLock(pVCpu->CTX_SUFF(pVM), &pVCpu->iem.s.aMemMappingLocks[iMemMap].Lock);
8612 Assert(pVCpu->iem.s.cActiveMappings > 0);
8613 pVCpu->iem.s.cActiveMappings--;
8614 }
8615 }
8616}
8617
8618
8619/**
8620 * Fetches a data byte.
8621 *
8622 * @returns Strict VBox status code.
8623 * @param pVCpu The cross context virtual CPU structure of the calling thread.
8624 * @param pu8Dst Where to return the byte.
8625 * @param iSegReg The index of the segment register to use for
8626 * this access. The base and limits are checked.
8627 * @param GCPtrMem The address of the guest memory.
8628 */
8629IEM_STATIC VBOXSTRICTRC iemMemFetchDataU8(PVMCPU pVCpu, uint8_t *pu8Dst, uint8_t iSegReg, RTGCPTR GCPtrMem)
8630{
8631 /* The lazy approach for now... */
8632 uint8_t const *pu8Src;
8633 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu8Src, sizeof(*pu8Src), iSegReg, GCPtrMem, IEM_ACCESS_DATA_R);
8634 if (rc == VINF_SUCCESS)
8635 {
8636 *pu8Dst = *pu8Src;
8637 rc = iemMemCommitAndUnmap(pVCpu, (void *)pu8Src, IEM_ACCESS_DATA_R);
8638 }
8639 return rc;
8640}
8641
8642
8643#ifdef IEM_WITH_SETJMP
8644/**
8645 * Fetches a data byte, longjmp on error.
8646 *
8647 * @returns The byte.
8648 * @param pVCpu The cross context virtual CPU structure of the calling thread.
8649 * @param iSegReg The index of the segment register to use for
8650 * this access. The base and limits are checked.
8651 * @param GCPtrMem The address of the guest memory.
8652 */
8653DECL_NO_INLINE(IEM_STATIC, uint8_t) iemMemFetchDataU8Jmp(PVMCPU pVCpu, uint8_t iSegReg, RTGCPTR GCPtrMem)
8654{
8655 /* The lazy approach for now... */
8656 uint8_t const *pu8Src = (uint8_t const *)iemMemMapJmp(pVCpu, sizeof(*pu8Src), iSegReg, GCPtrMem, IEM_ACCESS_DATA_R);
8657 uint8_t const bRet = *pu8Src;
8658 iemMemCommitAndUnmapJmp(pVCpu, (void *)pu8Src, IEM_ACCESS_DATA_R);
8659 return bRet;
8660}
8661#endif /* IEM_WITH_SETJMP */
8662
8663
8664/**
8665 * Fetches a data word.
8666 *
8667 * @returns Strict VBox status code.
8668 * @param pVCpu The cross context virtual CPU structure of the calling thread.
8669 * @param pu16Dst Where to return the word.
8670 * @param iSegReg The index of the segment register to use for
8671 * this access. The base and limits are checked.
8672 * @param GCPtrMem The address of the guest memory.
8673 */
8674IEM_STATIC VBOXSTRICTRC iemMemFetchDataU16(PVMCPU pVCpu, uint16_t *pu16Dst, uint8_t iSegReg, RTGCPTR GCPtrMem)
8675{
8676 /* The lazy approach for now... */
8677 uint16_t const *pu16Src;
8678 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu16Src, sizeof(*pu16Src), iSegReg, GCPtrMem, IEM_ACCESS_DATA_R);
8679 if (rc == VINF_SUCCESS)
8680 {
8681 *pu16Dst = *pu16Src;
8682 rc = iemMemCommitAndUnmap(pVCpu, (void *)pu16Src, IEM_ACCESS_DATA_R);
8683 }
8684 return rc;
8685}
8686
8687
8688#ifdef IEM_WITH_SETJMP
8689/**
8690 * Fetches a data word, longjmp on error.
8691 *
8692 * @returns The word
8693 * @param pVCpu The cross context virtual CPU structure of the calling thread.
8694 * @param iSegReg The index of the segment register to use for
8695 * this access. The base and limits are checked.
8696 * @param GCPtrMem The address of the guest memory.
8697 */
8698DECL_NO_INLINE(IEM_STATIC, uint16_t) iemMemFetchDataU16Jmp(PVMCPU pVCpu, uint8_t iSegReg, RTGCPTR GCPtrMem)
8699{
8700 /* The lazy approach for now... */
8701 uint16_t const *pu16Src = (uint16_t const *)iemMemMapJmp(pVCpu, sizeof(*pu16Src), iSegReg, GCPtrMem, IEM_ACCESS_DATA_R);
8702 uint16_t const u16Ret = *pu16Src;
8703 iemMemCommitAndUnmapJmp(pVCpu, (void *)pu16Src, IEM_ACCESS_DATA_R);
8704 return u16Ret;
8705}
8706#endif
8707
8708
8709/**
8710 * Fetches a data dword.
8711 *
8712 * @returns Strict VBox status code.
8713 * @param pVCpu The cross context virtual CPU structure of the calling thread.
8714 * @param pu32Dst Where to return the dword.
8715 * @param iSegReg The index of the segment register to use for
8716 * this access. The base and limits are checked.
8717 * @param GCPtrMem The address of the guest memory.
8718 */
8719IEM_STATIC VBOXSTRICTRC iemMemFetchDataU32(PVMCPU pVCpu, uint32_t *pu32Dst, uint8_t iSegReg, RTGCPTR GCPtrMem)
8720{
8721 /* The lazy approach for now... */
8722 uint32_t const *pu32Src;
8723 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu32Src, sizeof(*pu32Src), iSegReg, GCPtrMem, IEM_ACCESS_DATA_R);
8724 if (rc == VINF_SUCCESS)
8725 {
8726 *pu32Dst = *pu32Src;
8727 rc = iemMemCommitAndUnmap(pVCpu, (void *)pu32Src, IEM_ACCESS_DATA_R);
8728 }
8729 return rc;
8730}
8731
8732
8733#ifdef IEM_WITH_SETJMP
8734
8735IEM_STATIC RTGCPTR iemMemApplySegmentToReadJmp(PVMCPU pVCpu, uint8_t iSegReg, size_t cbMem, RTGCPTR GCPtrMem)
8736{
8737 Assert(cbMem >= 1);
8738 Assert(iSegReg < X86_SREG_COUNT);
8739
8740 /*
8741 * 64-bit mode is simpler.
8742 */
8743 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
8744 {
8745 if (iSegReg >= X86_SREG_FS)
8746 {
8747 PCPUMSELREGHID pSel = iemSRegGetHid(pVCpu, iSegReg);
8748 GCPtrMem += pSel->u64Base;
8749 }
8750
8751 if (RT_LIKELY(X86_IS_CANONICAL(GCPtrMem) && X86_IS_CANONICAL(GCPtrMem + cbMem - 1)))
8752 return GCPtrMem;
8753 }
8754 /*
8755 * 16-bit and 32-bit segmentation.
8756 */
8757 else
8758 {
8759 PCPUMSELREGHID pSel = iemSRegGetHid(pVCpu, iSegReg);
8760 if ( (pSel->Attr.u & (X86DESCATTR_P | X86DESCATTR_UNUSABLE | X86_SEL_TYPE_CODE | X86_SEL_TYPE_DOWN))
8761 == X86DESCATTR_P /* data, expand up */
8762 || (pSel->Attr.u & (X86DESCATTR_P | X86DESCATTR_UNUSABLE | X86_SEL_TYPE_CODE | X86_SEL_TYPE_READ))
8763 == (X86DESCATTR_P | X86_SEL_TYPE_CODE | X86_SEL_TYPE_READ) /* code, read-only */ )
8764 {
8765 /* expand up */
8766 uint32_t GCPtrLast32 = (uint32_t)GCPtrMem + (uint32_t)cbMem;
8767 if (RT_LIKELY( GCPtrLast32 > pSel->u32Limit
8768 && GCPtrLast32 > (uint32_t)GCPtrMem))
8769 return (uint32_t)GCPtrMem + (uint32_t)pSel->u64Base;
8770 }
8771 else if ( (pSel->Attr.u & (X86DESCATTR_P | X86DESCATTR_UNUSABLE | X86_SEL_TYPE_CODE | X86_SEL_TYPE_DOWN))
8772 == (X86DESCATTR_P | X86_SEL_TYPE_DOWN) /* data, expand down */ )
8773 {
8774 /* expand down */
8775 uint32_t GCPtrLast32 = (uint32_t)GCPtrMem + (uint32_t)cbMem;
8776 if (RT_LIKELY( (uint32_t)GCPtrMem > pSel->u32Limit
8777 && GCPtrLast32 <= (pSel->Attr.n.u1DefBig ? UINT32_MAX : UINT32_C(0xffff))
8778 && GCPtrLast32 > (uint32_t)GCPtrMem))
8779 return (uint32_t)GCPtrMem + (uint32_t)pSel->u64Base;
8780 }
8781 else
8782 iemRaiseSelectorInvalidAccessJmp(pVCpu, iSegReg, IEM_ACCESS_DATA_R);
8783 iemRaiseSelectorBoundsJmp(pVCpu, iSegReg, IEM_ACCESS_DATA_R);
8784 }
8785 iemRaiseGeneralProtectionFault0Jmp(pVCpu);
8786}
8787
8788
8789IEM_STATIC RTGCPTR iemMemApplySegmentToWriteJmp(PVMCPU pVCpu, uint8_t iSegReg, size_t cbMem, RTGCPTR GCPtrMem)
8790{
8791 Assert(cbMem >= 1);
8792 Assert(iSegReg < X86_SREG_COUNT);
8793
8794 /*
8795 * 64-bit mode is simpler.
8796 */
8797 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
8798 {
8799 if (iSegReg >= X86_SREG_FS)
8800 {
8801 PCPUMSELREGHID pSel = iemSRegGetHid(pVCpu, iSegReg);
8802 GCPtrMem += pSel->u64Base;
8803 }
8804
8805 if (RT_LIKELY(X86_IS_CANONICAL(GCPtrMem) && X86_IS_CANONICAL(GCPtrMem + cbMem - 1)))
8806 return GCPtrMem;
8807 }
8808 /*
8809 * 16-bit and 32-bit segmentation.
8810 */
8811 else
8812 {
8813 PCPUMSELREGHID pSel = iemSRegGetHid(pVCpu, iSegReg);
8814 uint32_t const fRelevantAttrs = pSel->Attr.u & ( X86DESCATTR_P | X86DESCATTR_UNUSABLE
8815 | X86_SEL_TYPE_CODE | X86_SEL_TYPE_WRITE | X86_SEL_TYPE_DOWN);
8816 if (fRelevantAttrs == (X86DESCATTR_P | X86_SEL_TYPE_WRITE)) /* data, expand up */
8817 {
8818 /* expand up */
8819 uint32_t GCPtrLast32 = (uint32_t)GCPtrMem + (uint32_t)cbMem;
8820 if (RT_LIKELY( GCPtrLast32 > pSel->u32Limit
8821 && GCPtrLast32 > (uint32_t)GCPtrMem))
8822 return (uint32_t)GCPtrMem + (uint32_t)pSel->u64Base;
8823 }
8824 else if (fRelevantAttrs == (X86DESCATTR_P | X86_SEL_TYPE_WRITE | X86_SEL_TYPE_DOWN)) /* data, expand up */
8825 {
8826 /* expand down */
8827 uint32_t GCPtrLast32 = (uint32_t)GCPtrMem + (uint32_t)cbMem;
8828 if (RT_LIKELY( (uint32_t)GCPtrMem > pSel->u32Limit
8829 && GCPtrLast32 <= (pSel->Attr.n.u1DefBig ? UINT32_MAX : UINT32_C(0xffff))
8830 && GCPtrLast32 > (uint32_t)GCPtrMem))
8831 return (uint32_t)GCPtrMem + (uint32_t)pSel->u64Base;
8832 }
8833 else
8834 iemRaiseSelectorInvalidAccessJmp(pVCpu, iSegReg, IEM_ACCESS_DATA_W);
8835 iemRaiseSelectorBoundsJmp(pVCpu, iSegReg, IEM_ACCESS_DATA_W);
8836 }
8837 iemRaiseGeneralProtectionFault0Jmp(pVCpu);
8838}
8839
8840
8841/**
8842 * Fetches a data dword, longjmp on error, fallback/safe version.
8843 *
8844 * @returns The dword
8845 * @param pVCpu The cross context virtual CPU structure of the calling thread.
8846 * @param iSegReg The index of the segment register to use for
8847 * this access. The base and limits are checked.
8848 * @param GCPtrMem The address of the guest memory.
8849 */
8850IEM_STATIC uint32_t iemMemFetchDataU32SafeJmp(PVMCPU pVCpu, uint8_t iSegReg, RTGCPTR GCPtrMem)
8851{
8852 uint32_t const *pu32Src = (uint32_t const *)iemMemMapJmp(pVCpu, sizeof(*pu32Src), iSegReg, GCPtrMem, IEM_ACCESS_DATA_R);
8853 uint32_t const u32Ret = *pu32Src;
8854 iemMemCommitAndUnmapJmp(pVCpu, (void *)pu32Src, IEM_ACCESS_DATA_R);
8855 return u32Ret;
8856}
8857
8858
8859/**
8860 * Fetches a data dword, longjmp on error.
8861 *
8862 * @returns The dword
8863 * @param pVCpu The cross context virtual CPU structure of the calling thread.
8864 * @param iSegReg The index of the segment register to use for
8865 * this access. The base and limits are checked.
8866 * @param GCPtrMem The address of the guest memory.
8867 */
8868DECL_NO_INLINE(IEM_STATIC, uint32_t) iemMemFetchDataU32Jmp(PVMCPU pVCpu, uint8_t iSegReg, RTGCPTR GCPtrMem)
8869{
8870# ifdef IEM_WITH_DATA_TLB
8871 RTGCPTR GCPtrEff = iemMemApplySegmentToReadJmp(pVCpu, iSegReg, sizeof(uint32_t), GCPtrMem);
8872 if (RT_LIKELY((GCPtrEff & X86_PAGE_OFFSET_MASK) <= X86_PAGE_SIZE - sizeof(uint32_t)))
8873 {
8874 /// @todo more later.
8875 }
8876
8877 return iemMemFetchDataU32SafeJmp(pVCpu, iSegReg, GCPtrMem);
8878# else
8879 /* The lazy approach. */
8880 uint32_t const *pu32Src = (uint32_t const *)iemMemMapJmp(pVCpu, sizeof(*pu32Src), iSegReg, GCPtrMem, IEM_ACCESS_DATA_R);
8881 uint32_t const u32Ret = *pu32Src;
8882 iemMemCommitAndUnmapJmp(pVCpu, (void *)pu32Src, IEM_ACCESS_DATA_R);
8883 return u32Ret;
8884# endif
8885}
8886#endif
8887
8888
8889#ifdef SOME_UNUSED_FUNCTION
8890/**
8891 * Fetches a data dword and sign extends it to a qword.
8892 *
8893 * @returns Strict VBox status code.
8894 * @param pVCpu The cross context virtual CPU structure of the calling thread.
8895 * @param pu64Dst Where to return the sign extended value.
8896 * @param iSegReg The index of the segment register to use for
8897 * this access. The base and limits are checked.
8898 * @param GCPtrMem The address of the guest memory.
8899 */
8900IEM_STATIC VBOXSTRICTRC iemMemFetchDataS32SxU64(PVMCPU pVCpu, uint64_t *pu64Dst, uint8_t iSegReg, RTGCPTR GCPtrMem)
8901{
8902 /* The lazy approach for now... */
8903 int32_t const *pi32Src;
8904 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pi32Src, sizeof(*pi32Src), iSegReg, GCPtrMem, IEM_ACCESS_DATA_R);
8905 if (rc == VINF_SUCCESS)
8906 {
8907 *pu64Dst = *pi32Src;
8908 rc = iemMemCommitAndUnmap(pVCpu, (void *)pi32Src, IEM_ACCESS_DATA_R);
8909 }
8910#ifdef __GNUC__ /* warning: GCC may be a royal pain */
8911 else
8912 *pu64Dst = 0;
8913#endif
8914 return rc;
8915}
8916#endif
8917
8918
8919/**
8920 * Fetches a data qword.
8921 *
8922 * @returns Strict VBox status code.
8923 * @param pVCpu The cross context virtual CPU structure of the calling thread.
8924 * @param pu64Dst Where to return the qword.
8925 * @param iSegReg The index of the segment register to use for
8926 * this access. The base and limits are checked.
8927 * @param GCPtrMem The address of the guest memory.
8928 */
8929IEM_STATIC VBOXSTRICTRC iemMemFetchDataU64(PVMCPU pVCpu, uint64_t *pu64Dst, uint8_t iSegReg, RTGCPTR GCPtrMem)
8930{
8931 /* The lazy approach for now... */
8932 uint64_t const *pu64Src;
8933 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu64Src, sizeof(*pu64Src), iSegReg, GCPtrMem, IEM_ACCESS_DATA_R);
8934 if (rc == VINF_SUCCESS)
8935 {
8936 *pu64Dst = *pu64Src;
8937 rc = iemMemCommitAndUnmap(pVCpu, (void *)pu64Src, IEM_ACCESS_DATA_R);
8938 }
8939 return rc;
8940}
8941
8942
8943#ifdef IEM_WITH_SETJMP
8944/**
8945 * Fetches a data qword, longjmp on error.
8946 *
8947 * @returns The qword.
8948 * @param pVCpu The cross context virtual CPU structure of the calling thread.
8949 * @param iSegReg The index of the segment register to use for
8950 * this access. The base and limits are checked.
8951 * @param GCPtrMem The address of the guest memory.
8952 */
8953DECL_NO_INLINE(IEM_STATIC, uint64_t) iemMemFetchDataU64Jmp(PVMCPU pVCpu, uint8_t iSegReg, RTGCPTR GCPtrMem)
8954{
8955 /* The lazy approach for now... */
8956 uint64_t const *pu64Src = (uint64_t const *)iemMemMapJmp(pVCpu, sizeof(*pu64Src), iSegReg, GCPtrMem, IEM_ACCESS_DATA_R);
8957 uint64_t const u64Ret = *pu64Src;
8958 iemMemCommitAndUnmapJmp(pVCpu, (void *)pu64Src, IEM_ACCESS_DATA_R);
8959 return u64Ret;
8960}
8961#endif
8962
8963
8964/**
8965 * Fetches a data qword, aligned at a 16 byte boundrary (for SSE).
8966 *
8967 * @returns Strict VBox status code.
8968 * @param pVCpu The cross context virtual CPU structure of the calling thread.
8969 * @param pu64Dst Where to return the qword.
8970 * @param iSegReg The index of the segment register to use for
8971 * this access. The base and limits are checked.
8972 * @param GCPtrMem The address of the guest memory.
8973 */
8974IEM_STATIC VBOXSTRICTRC iemMemFetchDataU64AlignedU128(PVMCPU pVCpu, uint64_t *pu64Dst, uint8_t iSegReg, RTGCPTR GCPtrMem)
8975{
8976 /* The lazy approach for now... */
8977 /** @todo testcase: Ordering of \#SS(0) vs \#GP() vs \#PF on SSE stuff. */
8978 if (RT_UNLIKELY(GCPtrMem & 15))
8979 return iemRaiseGeneralProtectionFault0(pVCpu);
8980
8981 uint64_t const *pu64Src;
8982 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu64Src, sizeof(*pu64Src), iSegReg, GCPtrMem, IEM_ACCESS_DATA_R);
8983 if (rc == VINF_SUCCESS)
8984 {
8985 *pu64Dst = *pu64Src;
8986 rc = iemMemCommitAndUnmap(pVCpu, (void *)pu64Src, IEM_ACCESS_DATA_R);
8987 }
8988 return rc;
8989}
8990
8991
8992#ifdef IEM_WITH_SETJMP
8993/**
8994 * Fetches a data qword, longjmp on error.
8995 *
8996 * @returns The qword.
8997 * @param pVCpu The cross context virtual CPU structure of the calling thread.
8998 * @param iSegReg The index of the segment register to use for
8999 * this access. The base and limits are checked.
9000 * @param GCPtrMem The address of the guest memory.
9001 */
9002DECL_NO_INLINE(IEM_STATIC, uint64_t) iemMemFetchDataU64AlignedU128Jmp(PVMCPU pVCpu, uint8_t iSegReg, RTGCPTR GCPtrMem)
9003{
9004 /* The lazy approach for now... */
9005 /** @todo testcase: Ordering of \#SS(0) vs \#GP() vs \#PF on SSE stuff. */
9006 if (RT_LIKELY(!(GCPtrMem & 15)))
9007 {
9008 uint64_t const *pu64Src = (uint64_t const *)iemMemMapJmp(pVCpu, sizeof(*pu64Src), iSegReg, GCPtrMem, IEM_ACCESS_DATA_R);
9009 uint64_t const u64Ret = *pu64Src;
9010 iemMemCommitAndUnmapJmp(pVCpu, (void *)pu64Src, IEM_ACCESS_DATA_R);
9011 return u64Ret;
9012 }
9013
9014 VBOXSTRICTRC rc = iemRaiseGeneralProtectionFault0(pVCpu);
9015 longjmp(*pVCpu->iem.s.CTX_SUFF(pJmpBuf), VBOXSTRICTRC_VAL(rc));
9016}
9017#endif
9018
9019
9020/**
9021 * Fetches a data tword.
9022 *
9023 * @returns Strict VBox status code.
9024 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9025 * @param pr80Dst Where to return the tword.
9026 * @param iSegReg The index of the segment register to use for
9027 * this access. The base and limits are checked.
9028 * @param GCPtrMem The address of the guest memory.
9029 */
9030IEM_STATIC VBOXSTRICTRC iemMemFetchDataR80(PVMCPU pVCpu, PRTFLOAT80U pr80Dst, uint8_t iSegReg, RTGCPTR GCPtrMem)
9031{
9032 /* The lazy approach for now... */
9033 PCRTFLOAT80U pr80Src;
9034 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pr80Src, sizeof(*pr80Src), iSegReg, GCPtrMem, IEM_ACCESS_DATA_R);
9035 if (rc == VINF_SUCCESS)
9036 {
9037 *pr80Dst = *pr80Src;
9038 rc = iemMemCommitAndUnmap(pVCpu, (void *)pr80Src, IEM_ACCESS_DATA_R);
9039 }
9040 return rc;
9041}
9042
9043
9044#ifdef IEM_WITH_SETJMP
9045/**
9046 * Fetches a data tword, longjmp on error.
9047 *
9048 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9049 * @param pr80Dst Where to return the tword.
9050 * @param iSegReg The index of the segment register to use for
9051 * this access. The base and limits are checked.
9052 * @param GCPtrMem The address of the guest memory.
9053 */
9054DECL_NO_INLINE(IEM_STATIC, void) iemMemFetchDataR80Jmp(PVMCPU pVCpu, PRTFLOAT80U pr80Dst, uint8_t iSegReg, RTGCPTR GCPtrMem)
9055{
9056 /* The lazy approach for now... */
9057 PCRTFLOAT80U pr80Src = (PCRTFLOAT80U)iemMemMapJmp(pVCpu, sizeof(*pr80Src), iSegReg, GCPtrMem, IEM_ACCESS_DATA_R);
9058 *pr80Dst = *pr80Src;
9059 iemMemCommitAndUnmapJmp(pVCpu, (void *)pr80Src, IEM_ACCESS_DATA_R);
9060}
9061#endif
9062
9063
9064/**
9065 * Fetches a data dqword (double qword), generally SSE related.
9066 *
9067 * @returns Strict VBox status code.
9068 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9069 * @param pu128Dst Where to return the qword.
9070 * @param iSegReg The index of the segment register to use for
9071 * this access. The base and limits are checked.
9072 * @param GCPtrMem The address of the guest memory.
9073 */
9074IEM_STATIC VBOXSTRICTRC iemMemFetchDataU128(PVMCPU pVCpu, uint128_t *pu128Dst, uint8_t iSegReg, RTGCPTR GCPtrMem)
9075{
9076 /* The lazy approach for now... */
9077 uint128_t const *pu128Src;
9078 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu128Src, sizeof(*pu128Src), iSegReg, GCPtrMem, IEM_ACCESS_DATA_R);
9079 if (rc == VINF_SUCCESS)
9080 {
9081 *pu128Dst = *pu128Src;
9082 rc = iemMemCommitAndUnmap(pVCpu, (void *)pu128Src, IEM_ACCESS_DATA_R);
9083 }
9084 return rc;
9085}
9086
9087
9088#ifdef IEM_WITH_SETJMP
9089/**
9090 * Fetches a data dqword (double qword), generally SSE related.
9091 *
9092 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9093 * @param pu128Dst Where to return the qword.
9094 * @param iSegReg The index of the segment register to use for
9095 * this access. The base and limits are checked.
9096 * @param GCPtrMem The address of the guest memory.
9097 */
9098IEM_STATIC void iemMemFetchDataU128Jmp(PVMCPU pVCpu, uint128_t *pu128Dst, uint8_t iSegReg, RTGCPTR GCPtrMem)
9099{
9100 /* The lazy approach for now... */
9101 uint128_t const *pu128Src = (uint128_t const *)iemMemMapJmp(pVCpu, sizeof(*pu128Src), iSegReg, GCPtrMem, IEM_ACCESS_DATA_R);
9102 *pu128Dst = *pu128Src;
9103 iemMemCommitAndUnmapJmp(pVCpu, (void *)pu128Src, IEM_ACCESS_DATA_R);
9104}
9105#endif
9106
9107
9108/**
9109 * Fetches a data dqword (double qword) at an aligned address, generally SSE
9110 * related.
9111 *
9112 * Raises \#GP(0) if not aligned.
9113 *
9114 * @returns Strict VBox status code.
9115 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9116 * @param pu128Dst Where to return the qword.
9117 * @param iSegReg The index of the segment register to use for
9118 * this access. The base and limits are checked.
9119 * @param GCPtrMem The address of the guest memory.
9120 */
9121IEM_STATIC VBOXSTRICTRC iemMemFetchDataU128AlignedSse(PVMCPU pVCpu, uint128_t *pu128Dst, uint8_t iSegReg, RTGCPTR GCPtrMem)
9122{
9123 /* The lazy approach for now... */
9124 /** @todo testcase: Ordering of \#SS(0) vs \#GP() vs \#PF on SSE stuff. */
9125 if ( (GCPtrMem & 15)
9126 && !(IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.MXCSR & X86_MXSCR_MM)) /** @todo should probably check this *after* applying seg.u64Base... Check real HW. */
9127 return iemRaiseGeneralProtectionFault0(pVCpu);
9128
9129 uint128_t const *pu128Src;
9130 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu128Src, sizeof(*pu128Src), iSegReg, GCPtrMem, IEM_ACCESS_DATA_R);
9131 if (rc == VINF_SUCCESS)
9132 {
9133 *pu128Dst = *pu128Src;
9134 rc = iemMemCommitAndUnmap(pVCpu, (void *)pu128Src, IEM_ACCESS_DATA_R);
9135 }
9136 return rc;
9137}
9138
9139
9140#ifdef IEM_WITH_SETJMP
9141/**
9142 * Fetches a data dqword (double qword) at an aligned address, generally SSE
9143 * related, longjmp on error.
9144 *
9145 * Raises \#GP(0) if not aligned.
9146 *
9147 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9148 * @param pu128Dst Where to return the qword.
9149 * @param iSegReg The index of the segment register to use for
9150 * this access. The base and limits are checked.
9151 * @param GCPtrMem The address of the guest memory.
9152 */
9153DECL_NO_INLINE(IEM_STATIC, void) iemMemFetchDataU128AlignedSseJmp(PVMCPU pVCpu, uint128_t *pu128Dst, uint8_t iSegReg, RTGCPTR GCPtrMem)
9154{
9155 /* The lazy approach for now... */
9156 /** @todo testcase: Ordering of \#SS(0) vs \#GP() vs \#PF on SSE stuff. */
9157 if ( (GCPtrMem & 15) == 0
9158 || (IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.MXCSR & X86_MXSCR_MM)) /** @todo should probably check this *after* applying seg.u64Base... Check real HW. */
9159 {
9160 uint128_t const *pu128Src = (uint128_t const *)iemMemMapJmp(pVCpu, sizeof(*pu128Src), iSegReg, GCPtrMem,
9161 IEM_ACCESS_DATA_R);
9162 *pu128Dst = *pu128Src;
9163 iemMemCommitAndUnmapJmp(pVCpu, (void *)pu128Src, IEM_ACCESS_DATA_R);
9164 return;
9165 }
9166
9167 VBOXSTRICTRC rcStrict = iemRaiseGeneralProtectionFault0(pVCpu);
9168 longjmp(*pVCpu->iem.s.CTX_SUFF(pJmpBuf), VBOXSTRICTRC_VAL(rcStrict));
9169}
9170#endif
9171
9172
9173
9174/**
9175 * Fetches a descriptor register (lgdt, lidt).
9176 *
9177 * @returns Strict VBox status code.
9178 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9179 * @param pcbLimit Where to return the limit.
9180 * @param pGCPtrBase Where to return the base.
9181 * @param iSegReg The index of the segment register to use for
9182 * this access. The base and limits are checked.
9183 * @param GCPtrMem The address of the guest memory.
9184 * @param enmOpSize The effective operand size.
9185 */
9186IEM_STATIC VBOXSTRICTRC iemMemFetchDataXdtr(PVMCPU pVCpu, uint16_t *pcbLimit, PRTGCPTR pGCPtrBase, uint8_t iSegReg,
9187 RTGCPTR GCPtrMem, IEMMODE enmOpSize)
9188{
9189 /*
9190 * Just like SIDT and SGDT, the LIDT and LGDT instructions are a
9191 * little special:
9192 * - The two reads are done separately.
9193 * - Operand size override works in 16-bit and 32-bit code, but 64-bit.
9194 * - We suspect the 386 to actually commit the limit before the base in
9195 * some cases (search for 386 in bs3CpuBasic2_lidt_lgdt_One). We
9196 * don't try emulate this eccentric behavior, because it's not well
9197 * enough understood and rather hard to trigger.
9198 * - The 486 seems to do a dword limit read when the operand size is 32-bit.
9199 */
9200 VBOXSTRICTRC rcStrict;
9201 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
9202 {
9203 rcStrict = iemMemFetchDataU16(pVCpu, pcbLimit, iSegReg, GCPtrMem);
9204 if (rcStrict == VINF_SUCCESS)
9205 rcStrict = iemMemFetchDataU64(pVCpu, pGCPtrBase, iSegReg, GCPtrMem + 2);
9206 }
9207 else
9208 {
9209 uint32_t uTmp = 0; /* (Visual C++ maybe used uninitialized) */
9210 if (enmOpSize == IEMMODE_32BIT)
9211 {
9212 if (IEM_GET_TARGET_CPU(pVCpu) != IEMTARGETCPU_486)
9213 {
9214 rcStrict = iemMemFetchDataU16(pVCpu, pcbLimit, iSegReg, GCPtrMem);
9215 if (rcStrict == VINF_SUCCESS)
9216 rcStrict = iemMemFetchDataU32(pVCpu, &uTmp, iSegReg, GCPtrMem + 2);
9217 }
9218 else
9219 {
9220 rcStrict = iemMemFetchDataU32(pVCpu, &uTmp, iSegReg, GCPtrMem);
9221 if (rcStrict == VINF_SUCCESS)
9222 {
9223 *pcbLimit = (uint16_t)uTmp;
9224 rcStrict = iemMemFetchDataU32(pVCpu, &uTmp, iSegReg, GCPtrMem + 2);
9225 }
9226 }
9227 if (rcStrict == VINF_SUCCESS)
9228 *pGCPtrBase = uTmp;
9229 }
9230 else
9231 {
9232 rcStrict = iemMemFetchDataU16(pVCpu, pcbLimit, iSegReg, GCPtrMem);
9233 if (rcStrict == VINF_SUCCESS)
9234 {
9235 rcStrict = iemMemFetchDataU32(pVCpu, &uTmp, iSegReg, GCPtrMem + 2);
9236 if (rcStrict == VINF_SUCCESS)
9237 *pGCPtrBase = uTmp & UINT32_C(0x00ffffff);
9238 }
9239 }
9240 }
9241 return rcStrict;
9242}
9243
9244
9245
9246/**
9247 * Stores a data byte.
9248 *
9249 * @returns Strict VBox status code.
9250 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9251 * @param iSegReg The index of the segment register to use for
9252 * this access. The base and limits are checked.
9253 * @param GCPtrMem The address of the guest memory.
9254 * @param u8Value The value to store.
9255 */
9256IEM_STATIC VBOXSTRICTRC iemMemStoreDataU8(PVMCPU pVCpu, uint8_t iSegReg, RTGCPTR GCPtrMem, uint8_t u8Value)
9257{
9258 /* The lazy approach for now... */
9259 uint8_t *pu8Dst;
9260 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu8Dst, sizeof(*pu8Dst), iSegReg, GCPtrMem, IEM_ACCESS_DATA_W);
9261 if (rc == VINF_SUCCESS)
9262 {
9263 *pu8Dst = u8Value;
9264 rc = iemMemCommitAndUnmap(pVCpu, pu8Dst, IEM_ACCESS_DATA_W);
9265 }
9266 return rc;
9267}
9268
9269
9270#ifdef IEM_WITH_SETJMP
9271/**
9272 * Stores a data byte, longjmp on error.
9273 *
9274 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9275 * @param iSegReg The index of the segment register to use for
9276 * this access. The base and limits are checked.
9277 * @param GCPtrMem The address of the guest memory.
9278 * @param u8Value The value to store.
9279 */
9280IEM_STATIC void iemMemStoreDataU8Jmp(PVMCPU pVCpu, uint8_t iSegReg, RTGCPTR GCPtrMem, uint8_t u8Value)
9281{
9282 /* The lazy approach for now... */
9283 uint8_t *pu8Dst = (uint8_t *)iemMemMapJmp(pVCpu, sizeof(*pu8Dst), iSegReg, GCPtrMem, IEM_ACCESS_DATA_W);
9284 *pu8Dst = u8Value;
9285 iemMemCommitAndUnmapJmp(pVCpu, pu8Dst, IEM_ACCESS_DATA_W);
9286}
9287#endif
9288
9289
9290/**
9291 * Stores a data word.
9292 *
9293 * @returns Strict VBox status code.
9294 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9295 * @param iSegReg The index of the segment register to use for
9296 * this access. The base and limits are checked.
9297 * @param GCPtrMem The address of the guest memory.
9298 * @param u16Value The value to store.
9299 */
9300IEM_STATIC VBOXSTRICTRC iemMemStoreDataU16(PVMCPU pVCpu, uint8_t iSegReg, RTGCPTR GCPtrMem, uint16_t u16Value)
9301{
9302 /* The lazy approach for now... */
9303 uint16_t *pu16Dst;
9304 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu16Dst, sizeof(*pu16Dst), iSegReg, GCPtrMem, IEM_ACCESS_DATA_W);
9305 if (rc == VINF_SUCCESS)
9306 {
9307 *pu16Dst = u16Value;
9308 rc = iemMemCommitAndUnmap(pVCpu, pu16Dst, IEM_ACCESS_DATA_W);
9309 }
9310 return rc;
9311}
9312
9313
9314#ifdef IEM_WITH_SETJMP
9315/**
9316 * Stores a data word, longjmp on error.
9317 *
9318 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9319 * @param iSegReg The index of the segment register to use for
9320 * this access. The base and limits are checked.
9321 * @param GCPtrMem The address of the guest memory.
9322 * @param u16Value The value to store.
9323 */
9324IEM_STATIC void iemMemStoreDataU16Jmp(PVMCPU pVCpu, uint8_t iSegReg, RTGCPTR GCPtrMem, uint16_t u16Value)
9325{
9326 /* The lazy approach for now... */
9327 uint16_t *pu16Dst = (uint16_t *)iemMemMapJmp(pVCpu, sizeof(*pu16Dst), iSegReg, GCPtrMem, IEM_ACCESS_DATA_W);
9328 *pu16Dst = u16Value;
9329 iemMemCommitAndUnmapJmp(pVCpu, pu16Dst, IEM_ACCESS_DATA_W);
9330}
9331#endif
9332
9333
9334/**
9335 * Stores a data dword.
9336 *
9337 * @returns Strict VBox status code.
9338 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9339 * @param iSegReg The index of the segment register to use for
9340 * this access. The base and limits are checked.
9341 * @param GCPtrMem The address of the guest memory.
9342 * @param u32Value The value to store.
9343 */
9344IEM_STATIC VBOXSTRICTRC iemMemStoreDataU32(PVMCPU pVCpu, uint8_t iSegReg, RTGCPTR GCPtrMem, uint32_t u32Value)
9345{
9346 /* The lazy approach for now... */
9347 uint32_t *pu32Dst;
9348 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu32Dst, sizeof(*pu32Dst), iSegReg, GCPtrMem, IEM_ACCESS_DATA_W);
9349 if (rc == VINF_SUCCESS)
9350 {
9351 *pu32Dst = u32Value;
9352 rc = iemMemCommitAndUnmap(pVCpu, pu32Dst, IEM_ACCESS_DATA_W);
9353 }
9354 return rc;
9355}
9356
9357
9358#ifdef IEM_WITH_SETJMP
9359/**
9360 * Stores a data dword.
9361 *
9362 * @returns Strict VBox status code.
9363 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9364 * @param iSegReg The index of the segment register to use for
9365 * this access. The base and limits are checked.
9366 * @param GCPtrMem The address of the guest memory.
9367 * @param u32Value The value to store.
9368 */
9369IEM_STATIC void iemMemStoreDataU32Jmp(PVMCPU pVCpu, uint8_t iSegReg, RTGCPTR GCPtrMem, uint32_t u32Value)
9370{
9371 /* The lazy approach for now... */
9372 uint32_t *pu32Dst = (uint32_t *)iemMemMapJmp(pVCpu, sizeof(*pu32Dst), iSegReg, GCPtrMem, IEM_ACCESS_DATA_W);
9373 *pu32Dst = u32Value;
9374 iemMemCommitAndUnmapJmp(pVCpu, pu32Dst, IEM_ACCESS_DATA_W);
9375}
9376#endif
9377
9378
9379/**
9380 * Stores a data qword.
9381 *
9382 * @returns Strict VBox status code.
9383 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9384 * @param iSegReg The index of the segment register to use for
9385 * this access. The base and limits are checked.
9386 * @param GCPtrMem The address of the guest memory.
9387 * @param u64Value The value to store.
9388 */
9389IEM_STATIC VBOXSTRICTRC iemMemStoreDataU64(PVMCPU pVCpu, uint8_t iSegReg, RTGCPTR GCPtrMem, uint64_t u64Value)
9390{
9391 /* The lazy approach for now... */
9392 uint64_t *pu64Dst;
9393 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu64Dst, sizeof(*pu64Dst), iSegReg, GCPtrMem, IEM_ACCESS_DATA_W);
9394 if (rc == VINF_SUCCESS)
9395 {
9396 *pu64Dst = u64Value;
9397 rc = iemMemCommitAndUnmap(pVCpu, pu64Dst, IEM_ACCESS_DATA_W);
9398 }
9399 return rc;
9400}
9401
9402
9403#ifdef IEM_WITH_SETJMP
9404/**
9405 * Stores a data qword, longjmp on error.
9406 *
9407 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9408 * @param iSegReg The index of the segment register to use for
9409 * this access. The base and limits are checked.
9410 * @param GCPtrMem The address of the guest memory.
9411 * @param u64Value The value to store.
9412 */
9413IEM_STATIC void iemMemStoreDataU64Jmp(PVMCPU pVCpu, uint8_t iSegReg, RTGCPTR GCPtrMem, uint64_t u64Value)
9414{
9415 /* The lazy approach for now... */
9416 uint64_t *pu64Dst = (uint64_t *)iemMemMapJmp(pVCpu, sizeof(*pu64Dst), iSegReg, GCPtrMem, IEM_ACCESS_DATA_W);
9417 *pu64Dst = u64Value;
9418 iemMemCommitAndUnmapJmp(pVCpu, pu64Dst, IEM_ACCESS_DATA_W);
9419}
9420#endif
9421
9422
9423/**
9424 * Stores a data dqword.
9425 *
9426 * @returns Strict VBox status code.
9427 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9428 * @param iSegReg The index of the segment register to use for
9429 * this access. The base and limits are checked.
9430 * @param GCPtrMem The address of the guest memory.
9431 * @param u128Value The value to store.
9432 */
9433IEM_STATIC VBOXSTRICTRC iemMemStoreDataU128(PVMCPU pVCpu, uint8_t iSegReg, RTGCPTR GCPtrMem, uint128_t u128Value)
9434{
9435 /* The lazy approach for now... */
9436 uint128_t *pu128Dst;
9437 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu128Dst, sizeof(*pu128Dst), iSegReg, GCPtrMem, IEM_ACCESS_DATA_W);
9438 if (rc == VINF_SUCCESS)
9439 {
9440 *pu128Dst = u128Value;
9441 rc = iemMemCommitAndUnmap(pVCpu, pu128Dst, IEM_ACCESS_DATA_W);
9442 }
9443 return rc;
9444}
9445
9446
9447#ifdef IEM_WITH_SETJMP
9448/**
9449 * Stores a data dqword, longjmp on error.
9450 *
9451 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9452 * @param iSegReg The index of the segment register to use for
9453 * this access. The base and limits are checked.
9454 * @param GCPtrMem The address of the guest memory.
9455 * @param u128Value The value to store.
9456 */
9457IEM_STATIC void iemMemStoreDataU128Jmp(PVMCPU pVCpu, uint8_t iSegReg, RTGCPTR GCPtrMem, uint128_t u128Value)
9458{
9459 /* The lazy approach for now... */
9460 uint128_t *pu128Dst = (uint128_t *)iemMemMapJmp(pVCpu, sizeof(*pu128Dst), iSegReg, GCPtrMem, IEM_ACCESS_DATA_W);
9461 *pu128Dst = u128Value;
9462 iemMemCommitAndUnmapJmp(pVCpu, pu128Dst, IEM_ACCESS_DATA_W);
9463}
9464#endif
9465
9466
9467/**
9468 * Stores a data dqword, SSE aligned.
9469 *
9470 * @returns Strict VBox status code.
9471 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9472 * @param iSegReg The index of the segment register to use for
9473 * this access. The base and limits are checked.
9474 * @param GCPtrMem The address of the guest memory.
9475 * @param u128Value The value to store.
9476 */
9477IEM_STATIC VBOXSTRICTRC iemMemStoreDataU128AlignedSse(PVMCPU pVCpu, uint8_t iSegReg, RTGCPTR GCPtrMem, uint128_t u128Value)
9478{
9479 /* The lazy approach for now... */
9480 if ( (GCPtrMem & 15)
9481 && !(IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.MXCSR & X86_MXSCR_MM)) /** @todo should probably check this *after* applying seg.u64Base... Check real HW. */
9482 return iemRaiseGeneralProtectionFault0(pVCpu);
9483
9484 uint128_t *pu128Dst;
9485 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu128Dst, sizeof(*pu128Dst), iSegReg, GCPtrMem, IEM_ACCESS_DATA_W);
9486 if (rc == VINF_SUCCESS)
9487 {
9488 *pu128Dst = u128Value;
9489 rc = iemMemCommitAndUnmap(pVCpu, pu128Dst, IEM_ACCESS_DATA_W);
9490 }
9491 return rc;
9492}
9493
9494
9495#ifdef IEM_WITH_SETJMP
9496/**
9497 * Stores a data dqword, SSE aligned.
9498 *
9499 * @returns Strict VBox status code.
9500 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9501 * @param iSegReg The index of the segment register to use for
9502 * this access. The base and limits are checked.
9503 * @param GCPtrMem The address of the guest memory.
9504 * @param u128Value The value to store.
9505 */
9506DECL_NO_INLINE(IEM_STATIC, void)
9507iemMemStoreDataU128AlignedSseJmp(PVMCPU pVCpu, uint8_t iSegReg, RTGCPTR GCPtrMem, uint128_t u128Value)
9508{
9509 /* The lazy approach for now... */
9510 if ( (GCPtrMem & 15) == 0
9511 || (IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.MXCSR & X86_MXSCR_MM)) /** @todo should probably check this *after* applying seg.u64Base... Check real HW. */
9512 {
9513 uint128_t *pu128Dst = (uint128_t *)iemMemMapJmp(pVCpu, sizeof(*pu128Dst), iSegReg, GCPtrMem, IEM_ACCESS_DATA_W);
9514 *pu128Dst = u128Value;
9515 iemMemCommitAndUnmapJmp(pVCpu, pu128Dst, IEM_ACCESS_DATA_W);
9516 return;
9517 }
9518
9519 VBOXSTRICTRC rcStrict = iemRaiseGeneralProtectionFault0(pVCpu);
9520 longjmp(*pVCpu->iem.s.CTX_SUFF(pJmpBuf), VBOXSTRICTRC_VAL(rcStrict));
9521}
9522#endif
9523
9524
9525/**
9526 * Stores a descriptor register (sgdt, sidt).
9527 *
9528 * @returns Strict VBox status code.
9529 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9530 * @param cbLimit The limit.
9531 * @param GCPtrBase The base address.
9532 * @param iSegReg The index of the segment register to use for
9533 * this access. The base and limits are checked.
9534 * @param GCPtrMem The address of the guest memory.
9535 */
9536IEM_STATIC VBOXSTRICTRC
9537iemMemStoreDataXdtr(PVMCPU pVCpu, uint16_t cbLimit, RTGCPTR GCPtrBase, uint8_t iSegReg, RTGCPTR GCPtrMem)
9538{
9539 /*
9540 * The SIDT and SGDT instructions actually stores the data using two
9541 * independent writes. The instructions does not respond to opsize prefixes.
9542 */
9543 VBOXSTRICTRC rcStrict = iemMemStoreDataU16(pVCpu, iSegReg, GCPtrMem, cbLimit);
9544 if (rcStrict == VINF_SUCCESS)
9545 {
9546 if (pVCpu->iem.s.enmCpuMode == IEMMODE_16BIT)
9547 rcStrict = iemMemStoreDataU32(pVCpu, iSegReg, GCPtrMem + 2,
9548 IEM_GET_TARGET_CPU(pVCpu) <= IEMTARGETCPU_286
9549 ? (uint32_t)GCPtrBase | UINT32_C(0xff000000) : (uint32_t)GCPtrBase);
9550 else if (pVCpu->iem.s.enmCpuMode == IEMMODE_32BIT)
9551 rcStrict = iemMemStoreDataU32(pVCpu, iSegReg, GCPtrMem + 2, (uint32_t)GCPtrBase);
9552 else
9553 rcStrict = iemMemStoreDataU64(pVCpu, iSegReg, GCPtrMem + 2, GCPtrBase);
9554 }
9555 return rcStrict;
9556}
9557
9558
9559/**
9560 * Pushes a word onto the stack.
9561 *
9562 * @returns Strict VBox status code.
9563 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9564 * @param u16Value The value to push.
9565 */
9566IEM_STATIC VBOXSTRICTRC iemMemStackPushU16(PVMCPU pVCpu, uint16_t u16Value)
9567{
9568 /* Increment the stack pointer. */
9569 uint64_t uNewRsp;
9570 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
9571 RTGCPTR GCPtrTop = iemRegGetRspForPush(pVCpu, pCtx, 2, &uNewRsp);
9572
9573 /* Write the word the lazy way. */
9574 uint16_t *pu16Dst;
9575 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu16Dst, sizeof(*pu16Dst), X86_SREG_SS, GCPtrTop, IEM_ACCESS_STACK_W);
9576 if (rc == VINF_SUCCESS)
9577 {
9578 *pu16Dst = u16Value;
9579 rc = iemMemCommitAndUnmap(pVCpu, pu16Dst, IEM_ACCESS_STACK_W);
9580 }
9581
9582 /* Commit the new RSP value unless we an access handler made trouble. */
9583 if (rc == VINF_SUCCESS)
9584 pCtx->rsp = uNewRsp;
9585
9586 return rc;
9587}
9588
9589
9590/**
9591 * Pushes a dword onto the stack.
9592 *
9593 * @returns Strict VBox status code.
9594 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9595 * @param u32Value The value to push.
9596 */
9597IEM_STATIC VBOXSTRICTRC iemMemStackPushU32(PVMCPU pVCpu, uint32_t u32Value)
9598{
9599 /* Increment the stack pointer. */
9600 uint64_t uNewRsp;
9601 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
9602 RTGCPTR GCPtrTop = iemRegGetRspForPush(pVCpu, pCtx, 4, &uNewRsp);
9603
9604 /* Write the dword the lazy way. */
9605 uint32_t *pu32Dst;
9606 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu32Dst, sizeof(*pu32Dst), X86_SREG_SS, GCPtrTop, IEM_ACCESS_STACK_W);
9607 if (rc == VINF_SUCCESS)
9608 {
9609 *pu32Dst = u32Value;
9610 rc = iemMemCommitAndUnmap(pVCpu, pu32Dst, IEM_ACCESS_STACK_W);
9611 }
9612
9613 /* Commit the new RSP value unless we an access handler made trouble. */
9614 if (rc == VINF_SUCCESS)
9615 pCtx->rsp = uNewRsp;
9616
9617 return rc;
9618}
9619
9620
9621/**
9622 * Pushes a dword segment register value onto the stack.
9623 *
9624 * @returns Strict VBox status code.
9625 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9626 * @param u32Value The value to push.
9627 */
9628IEM_STATIC VBOXSTRICTRC iemMemStackPushU32SReg(PVMCPU pVCpu, uint32_t u32Value)
9629{
9630 /* Increment the stack pointer. */
9631 uint64_t uNewRsp;
9632 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
9633 RTGCPTR GCPtrTop = iemRegGetRspForPush(pVCpu, pCtx, 4, &uNewRsp);
9634
9635 VBOXSTRICTRC rc;
9636 if (IEM_FULL_VERIFICATION_REM_ENABLED(pVCpu))
9637 {
9638 /* The recompiler writes a full dword. */
9639 uint32_t *pu32Dst;
9640 rc = iemMemMap(pVCpu, (void **)&pu32Dst, sizeof(*pu32Dst), X86_SREG_SS, GCPtrTop, IEM_ACCESS_STACK_W);
9641 if (rc == VINF_SUCCESS)
9642 {
9643 *pu32Dst = u32Value;
9644 rc = iemMemCommitAndUnmap(pVCpu, pu32Dst, IEM_ACCESS_STACK_W);
9645 }
9646 }
9647 else
9648 {
9649 /* The intel docs talks about zero extending the selector register
9650 value. My actual intel CPU here might be zero extending the value
9651 but it still only writes the lower word... */
9652 /** @todo Test this on new HW and on AMD and in 64-bit mode. Also test what
9653 * happens when crossing an electric page boundrary, is the high word checked
9654 * for write accessibility or not? Probably it is. What about segment limits?
9655 * It appears this behavior is also shared with trap error codes.
9656 *
9657 * Docs indicate the behavior changed maybe in Pentium or Pentium Pro. Check
9658 * ancient hardware when it actually did change. */
9659 uint16_t *pu16Dst;
9660 rc = iemMemMap(pVCpu, (void **)&pu16Dst, sizeof(uint32_t), X86_SREG_SS, GCPtrTop, IEM_ACCESS_STACK_RW);
9661 if (rc == VINF_SUCCESS)
9662 {
9663 *pu16Dst = (uint16_t)u32Value;
9664 rc = iemMemCommitAndUnmap(pVCpu, pu16Dst, IEM_ACCESS_STACK_RW);
9665 }
9666 }
9667
9668 /* Commit the new RSP value unless we an access handler made trouble. */
9669 if (rc == VINF_SUCCESS)
9670 pCtx->rsp = uNewRsp;
9671
9672 return rc;
9673}
9674
9675
9676/**
9677 * Pushes a qword onto the stack.
9678 *
9679 * @returns Strict VBox status code.
9680 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9681 * @param u64Value The value to push.
9682 */
9683IEM_STATIC VBOXSTRICTRC iemMemStackPushU64(PVMCPU pVCpu, uint64_t u64Value)
9684{
9685 /* Increment the stack pointer. */
9686 uint64_t uNewRsp;
9687 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
9688 RTGCPTR GCPtrTop = iemRegGetRspForPush(pVCpu, pCtx, 8, &uNewRsp);
9689
9690 /* Write the word the lazy way. */
9691 uint64_t *pu64Dst;
9692 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu64Dst, sizeof(*pu64Dst), X86_SREG_SS, GCPtrTop, IEM_ACCESS_STACK_W);
9693 if (rc == VINF_SUCCESS)
9694 {
9695 *pu64Dst = u64Value;
9696 rc = iemMemCommitAndUnmap(pVCpu, pu64Dst, IEM_ACCESS_STACK_W);
9697 }
9698
9699 /* Commit the new RSP value unless we an access handler made trouble. */
9700 if (rc == VINF_SUCCESS)
9701 pCtx->rsp = uNewRsp;
9702
9703 return rc;
9704}
9705
9706
9707/**
9708 * Pops a word from the stack.
9709 *
9710 * @returns Strict VBox status code.
9711 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9712 * @param pu16Value Where to store the popped value.
9713 */
9714IEM_STATIC VBOXSTRICTRC iemMemStackPopU16(PVMCPU pVCpu, uint16_t *pu16Value)
9715{
9716 /* Increment the stack pointer. */
9717 uint64_t uNewRsp;
9718 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
9719 RTGCPTR GCPtrTop = iemRegGetRspForPop(pVCpu, pCtx, 2, &uNewRsp);
9720
9721 /* Write the word the lazy way. */
9722 uint16_t const *pu16Src;
9723 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu16Src, sizeof(*pu16Src), X86_SREG_SS, GCPtrTop, IEM_ACCESS_STACK_R);
9724 if (rc == VINF_SUCCESS)
9725 {
9726 *pu16Value = *pu16Src;
9727 rc = iemMemCommitAndUnmap(pVCpu, (void *)pu16Src, IEM_ACCESS_STACK_R);
9728
9729 /* Commit the new RSP value. */
9730 if (rc == VINF_SUCCESS)
9731 pCtx->rsp = uNewRsp;
9732 }
9733
9734 return rc;
9735}
9736
9737
9738/**
9739 * Pops a dword from the stack.
9740 *
9741 * @returns Strict VBox status code.
9742 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9743 * @param pu32Value Where to store the popped value.
9744 */
9745IEM_STATIC VBOXSTRICTRC iemMemStackPopU32(PVMCPU pVCpu, uint32_t *pu32Value)
9746{
9747 /* Increment the stack pointer. */
9748 uint64_t uNewRsp;
9749 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
9750 RTGCPTR GCPtrTop = iemRegGetRspForPop(pVCpu, pCtx, 4, &uNewRsp);
9751
9752 /* Write the word the lazy way. */
9753 uint32_t const *pu32Src;
9754 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu32Src, sizeof(*pu32Src), X86_SREG_SS, GCPtrTop, IEM_ACCESS_STACK_R);
9755 if (rc == VINF_SUCCESS)
9756 {
9757 *pu32Value = *pu32Src;
9758 rc = iemMemCommitAndUnmap(pVCpu, (void *)pu32Src, IEM_ACCESS_STACK_R);
9759
9760 /* Commit the new RSP value. */
9761 if (rc == VINF_SUCCESS)
9762 pCtx->rsp = uNewRsp;
9763 }
9764
9765 return rc;
9766}
9767
9768
9769/**
9770 * Pops a qword from the stack.
9771 *
9772 * @returns Strict VBox status code.
9773 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9774 * @param pu64Value Where to store the popped value.
9775 */
9776IEM_STATIC VBOXSTRICTRC iemMemStackPopU64(PVMCPU pVCpu, uint64_t *pu64Value)
9777{
9778 /* Increment the stack pointer. */
9779 uint64_t uNewRsp;
9780 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
9781 RTGCPTR GCPtrTop = iemRegGetRspForPop(pVCpu, pCtx, 8, &uNewRsp);
9782
9783 /* Write the word the lazy way. */
9784 uint64_t const *pu64Src;
9785 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu64Src, sizeof(*pu64Src), X86_SREG_SS, GCPtrTop, IEM_ACCESS_STACK_R);
9786 if (rc == VINF_SUCCESS)
9787 {
9788 *pu64Value = *pu64Src;
9789 rc = iemMemCommitAndUnmap(pVCpu, (void *)pu64Src, IEM_ACCESS_STACK_R);
9790
9791 /* Commit the new RSP value. */
9792 if (rc == VINF_SUCCESS)
9793 pCtx->rsp = uNewRsp;
9794 }
9795
9796 return rc;
9797}
9798
9799
9800/**
9801 * Pushes a word onto the stack, using a temporary stack pointer.
9802 *
9803 * @returns Strict VBox status code.
9804 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9805 * @param u16Value The value to push.
9806 * @param pTmpRsp Pointer to the temporary stack pointer.
9807 */
9808IEM_STATIC VBOXSTRICTRC iemMemStackPushU16Ex(PVMCPU pVCpu, uint16_t u16Value, PRTUINT64U pTmpRsp)
9809{
9810 /* Increment the stack pointer. */
9811 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
9812 RTUINT64U NewRsp = *pTmpRsp;
9813 RTGCPTR GCPtrTop = iemRegGetRspForPushEx(pVCpu, pCtx, &NewRsp, 2);
9814
9815 /* Write the word the lazy way. */
9816 uint16_t *pu16Dst;
9817 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu16Dst, sizeof(*pu16Dst), X86_SREG_SS, GCPtrTop, IEM_ACCESS_STACK_W);
9818 if (rc == VINF_SUCCESS)
9819 {
9820 *pu16Dst = u16Value;
9821 rc = iemMemCommitAndUnmap(pVCpu, pu16Dst, IEM_ACCESS_STACK_W);
9822 }
9823
9824 /* Commit the new RSP value unless we an access handler made trouble. */
9825 if (rc == VINF_SUCCESS)
9826 *pTmpRsp = NewRsp;
9827
9828 return rc;
9829}
9830
9831
9832/**
9833 * Pushes a dword onto the stack, using a temporary stack pointer.
9834 *
9835 * @returns Strict VBox status code.
9836 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9837 * @param u32Value The value to push.
9838 * @param pTmpRsp Pointer to the temporary stack pointer.
9839 */
9840IEM_STATIC VBOXSTRICTRC iemMemStackPushU32Ex(PVMCPU pVCpu, uint32_t u32Value, PRTUINT64U pTmpRsp)
9841{
9842 /* Increment the stack pointer. */
9843 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
9844 RTUINT64U NewRsp = *pTmpRsp;
9845 RTGCPTR GCPtrTop = iemRegGetRspForPushEx(pVCpu, pCtx, &NewRsp, 4);
9846
9847 /* Write the word the lazy way. */
9848 uint32_t *pu32Dst;
9849 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu32Dst, sizeof(*pu32Dst), X86_SREG_SS, GCPtrTop, IEM_ACCESS_STACK_W);
9850 if (rc == VINF_SUCCESS)
9851 {
9852 *pu32Dst = u32Value;
9853 rc = iemMemCommitAndUnmap(pVCpu, pu32Dst, IEM_ACCESS_STACK_W);
9854 }
9855
9856 /* Commit the new RSP value unless we an access handler made trouble. */
9857 if (rc == VINF_SUCCESS)
9858 *pTmpRsp = NewRsp;
9859
9860 return rc;
9861}
9862
9863
9864/**
9865 * Pushes a dword onto the stack, using a temporary stack pointer.
9866 *
9867 * @returns Strict VBox status code.
9868 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9869 * @param u64Value The value to push.
9870 * @param pTmpRsp Pointer to the temporary stack pointer.
9871 */
9872IEM_STATIC VBOXSTRICTRC iemMemStackPushU64Ex(PVMCPU pVCpu, uint64_t u64Value, PRTUINT64U pTmpRsp)
9873{
9874 /* Increment the stack pointer. */
9875 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
9876 RTUINT64U NewRsp = *pTmpRsp;
9877 RTGCPTR GCPtrTop = iemRegGetRspForPushEx(pVCpu, pCtx, &NewRsp, 8);
9878
9879 /* Write the word the lazy way. */
9880 uint64_t *pu64Dst;
9881 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu64Dst, sizeof(*pu64Dst), X86_SREG_SS, GCPtrTop, IEM_ACCESS_STACK_W);
9882 if (rc == VINF_SUCCESS)
9883 {
9884 *pu64Dst = u64Value;
9885 rc = iemMemCommitAndUnmap(pVCpu, pu64Dst, IEM_ACCESS_STACK_W);
9886 }
9887
9888 /* Commit the new RSP value unless we an access handler made trouble. */
9889 if (rc == VINF_SUCCESS)
9890 *pTmpRsp = NewRsp;
9891
9892 return rc;
9893}
9894
9895
9896/**
9897 * Pops a word from the stack, using a temporary stack pointer.
9898 *
9899 * @returns Strict VBox status code.
9900 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9901 * @param pu16Value Where to store the popped value.
9902 * @param pTmpRsp Pointer to the temporary stack pointer.
9903 */
9904IEM_STATIC VBOXSTRICTRC iemMemStackPopU16Ex(PVMCPU pVCpu, uint16_t *pu16Value, PRTUINT64U pTmpRsp)
9905{
9906 /* Increment the stack pointer. */
9907 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
9908 RTUINT64U NewRsp = *pTmpRsp;
9909 RTGCPTR GCPtrTop = iemRegGetRspForPopEx(pVCpu, pCtx, &NewRsp, 2);
9910
9911 /* Write the word the lazy way. */
9912 uint16_t const *pu16Src;
9913 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu16Src, sizeof(*pu16Src), X86_SREG_SS, GCPtrTop, IEM_ACCESS_STACK_R);
9914 if (rc == VINF_SUCCESS)
9915 {
9916 *pu16Value = *pu16Src;
9917 rc = iemMemCommitAndUnmap(pVCpu, (void *)pu16Src, IEM_ACCESS_STACK_R);
9918
9919 /* Commit the new RSP value. */
9920 if (rc == VINF_SUCCESS)
9921 *pTmpRsp = NewRsp;
9922 }
9923
9924 return rc;
9925}
9926
9927
9928/**
9929 * Pops a dword from the stack, using a temporary stack pointer.
9930 *
9931 * @returns Strict VBox status code.
9932 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9933 * @param pu32Value Where to store the popped value.
9934 * @param pTmpRsp Pointer to the temporary stack pointer.
9935 */
9936IEM_STATIC VBOXSTRICTRC iemMemStackPopU32Ex(PVMCPU pVCpu, uint32_t *pu32Value, PRTUINT64U pTmpRsp)
9937{
9938 /* Increment the stack pointer. */
9939 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
9940 RTUINT64U NewRsp = *pTmpRsp;
9941 RTGCPTR GCPtrTop = iemRegGetRspForPopEx(pVCpu, pCtx, &NewRsp, 4);
9942
9943 /* Write the word the lazy way. */
9944 uint32_t const *pu32Src;
9945 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu32Src, sizeof(*pu32Src), X86_SREG_SS, GCPtrTop, IEM_ACCESS_STACK_R);
9946 if (rc == VINF_SUCCESS)
9947 {
9948 *pu32Value = *pu32Src;
9949 rc = iemMemCommitAndUnmap(pVCpu, (void *)pu32Src, IEM_ACCESS_STACK_R);
9950
9951 /* Commit the new RSP value. */
9952 if (rc == VINF_SUCCESS)
9953 *pTmpRsp = NewRsp;
9954 }
9955
9956 return rc;
9957}
9958
9959
9960/**
9961 * Pops a qword from the stack, using a temporary stack pointer.
9962 *
9963 * @returns Strict VBox status code.
9964 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9965 * @param pu64Value Where to store the popped value.
9966 * @param pTmpRsp Pointer to the temporary stack pointer.
9967 */
9968IEM_STATIC VBOXSTRICTRC iemMemStackPopU64Ex(PVMCPU pVCpu, uint64_t *pu64Value, PRTUINT64U pTmpRsp)
9969{
9970 /* Increment the stack pointer. */
9971 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
9972 RTUINT64U NewRsp = *pTmpRsp;
9973 RTGCPTR GCPtrTop = iemRegGetRspForPopEx(pVCpu, pCtx, &NewRsp, 8);
9974
9975 /* Write the word the lazy way. */
9976 uint64_t const *pu64Src;
9977 VBOXSTRICTRC rcStrict = iemMemMap(pVCpu, (void **)&pu64Src, sizeof(*pu64Src), X86_SREG_SS, GCPtrTop, IEM_ACCESS_STACK_R);
9978 if (rcStrict == VINF_SUCCESS)
9979 {
9980 *pu64Value = *pu64Src;
9981 rcStrict = iemMemCommitAndUnmap(pVCpu, (void *)pu64Src, IEM_ACCESS_STACK_R);
9982
9983 /* Commit the new RSP value. */
9984 if (rcStrict == VINF_SUCCESS)
9985 *pTmpRsp = NewRsp;
9986 }
9987
9988 return rcStrict;
9989}
9990
9991
9992/**
9993 * Begin a special stack push (used by interrupt, exceptions and such).
9994 *
9995 * This will raise \#SS or \#PF if appropriate.
9996 *
9997 * @returns Strict VBox status code.
9998 * @param pVCpu The cross context virtual CPU structure of the calling thread.
9999 * @param cbMem The number of bytes to push onto the stack.
10000 * @param ppvMem Where to return the pointer to the stack memory.
10001 * As with the other memory functions this could be
10002 * direct access or bounce buffered access, so
10003 * don't commit register until the commit call
10004 * succeeds.
10005 * @param puNewRsp Where to return the new RSP value. This must be
10006 * passed unchanged to
10007 * iemMemStackPushCommitSpecial().
10008 */
10009IEM_STATIC VBOXSTRICTRC iemMemStackPushBeginSpecial(PVMCPU pVCpu, size_t cbMem, void **ppvMem, uint64_t *puNewRsp)
10010{
10011 Assert(cbMem < UINT8_MAX);
10012 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
10013 RTGCPTR GCPtrTop = iemRegGetRspForPush(pVCpu, pCtx, (uint8_t)cbMem, puNewRsp);
10014 return iemMemMap(pVCpu, ppvMem, cbMem, X86_SREG_SS, GCPtrTop, IEM_ACCESS_STACK_W);
10015}
10016
10017
10018/**
10019 * Commits a special stack push (started by iemMemStackPushBeginSpecial).
10020 *
10021 * This will update the rSP.
10022 *
10023 * @returns Strict VBox status code.
10024 * @param pVCpu The cross context virtual CPU structure of the calling thread.
10025 * @param pvMem The pointer returned by
10026 * iemMemStackPushBeginSpecial().
10027 * @param uNewRsp The new RSP value returned by
10028 * iemMemStackPushBeginSpecial().
10029 */
10030IEM_STATIC VBOXSTRICTRC iemMemStackPushCommitSpecial(PVMCPU pVCpu, void *pvMem, uint64_t uNewRsp)
10031{
10032 VBOXSTRICTRC rcStrict = iemMemCommitAndUnmap(pVCpu, pvMem, IEM_ACCESS_STACK_W);
10033 if (rcStrict == VINF_SUCCESS)
10034 IEM_GET_CTX(pVCpu)->rsp = uNewRsp;
10035 return rcStrict;
10036}
10037
10038
10039/**
10040 * Begin a special stack pop (used by iret, retf and such).
10041 *
10042 * This will raise \#SS or \#PF if appropriate.
10043 *
10044 * @returns Strict VBox status code.
10045 * @param pVCpu The cross context virtual CPU structure of the calling thread.
10046 * @param cbMem The number of bytes to pop from the stack.
10047 * @param ppvMem Where to return the pointer to the stack memory.
10048 * @param puNewRsp Where to return the new RSP value. This must be
10049 * assigned to CPUMCTX::rsp manually some time
10050 * after iemMemStackPopDoneSpecial() has been
10051 * called.
10052 */
10053IEM_STATIC VBOXSTRICTRC iemMemStackPopBeginSpecial(PVMCPU pVCpu, size_t cbMem, void const **ppvMem, uint64_t *puNewRsp)
10054{
10055 Assert(cbMem < UINT8_MAX);
10056 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
10057 RTGCPTR GCPtrTop = iemRegGetRspForPop(pVCpu, pCtx, (uint8_t)cbMem, puNewRsp);
10058 return iemMemMap(pVCpu, (void **)ppvMem, cbMem, X86_SREG_SS, GCPtrTop, IEM_ACCESS_STACK_R);
10059}
10060
10061
10062/**
10063 * Continue a special stack pop (used by iret and retf).
10064 *
10065 * This will raise \#SS or \#PF if appropriate.
10066 *
10067 * @returns Strict VBox status code.
10068 * @param pVCpu The cross context virtual CPU structure of the calling thread.
10069 * @param cbMem The number of bytes to pop from the stack.
10070 * @param ppvMem Where to return the pointer to the stack memory.
10071 * @param puNewRsp Where to return the new RSP value. This must be
10072 * assigned to CPUMCTX::rsp manually some time
10073 * after iemMemStackPopDoneSpecial() has been
10074 * called.
10075 */
10076IEM_STATIC VBOXSTRICTRC iemMemStackPopContinueSpecial(PVMCPU pVCpu, size_t cbMem, void const **ppvMem, uint64_t *puNewRsp)
10077{
10078 Assert(cbMem < UINT8_MAX);
10079 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
10080 RTUINT64U NewRsp;
10081 NewRsp.u = *puNewRsp;
10082 RTGCPTR GCPtrTop = iemRegGetRspForPopEx(pVCpu, pCtx, &NewRsp, 8);
10083 *puNewRsp = NewRsp.u;
10084 return iemMemMap(pVCpu, (void **)ppvMem, cbMem, X86_SREG_SS, GCPtrTop, IEM_ACCESS_STACK_R);
10085}
10086
10087
10088/**
10089 * Done with a special stack pop (started by iemMemStackPopBeginSpecial or
10090 * iemMemStackPopContinueSpecial).
10091 *
10092 * The caller will manually commit the rSP.
10093 *
10094 * @returns Strict VBox status code.
10095 * @param pVCpu The cross context virtual CPU structure of the calling thread.
10096 * @param pvMem The pointer returned by
10097 * iemMemStackPopBeginSpecial() or
10098 * iemMemStackPopContinueSpecial().
10099 */
10100IEM_STATIC VBOXSTRICTRC iemMemStackPopDoneSpecial(PVMCPU pVCpu, void const *pvMem)
10101{
10102 return iemMemCommitAndUnmap(pVCpu, (void *)pvMem, IEM_ACCESS_STACK_R);
10103}
10104
10105
10106/**
10107 * Fetches a system table byte.
10108 *
10109 * @returns Strict VBox status code.
10110 * @param pVCpu The cross context virtual CPU structure of the calling thread.
10111 * @param pbDst Where to return the byte.
10112 * @param iSegReg The index of the segment register to use for
10113 * this access. The base and limits are checked.
10114 * @param GCPtrMem The address of the guest memory.
10115 */
10116IEM_STATIC VBOXSTRICTRC iemMemFetchSysU8(PVMCPU pVCpu, uint8_t *pbDst, uint8_t iSegReg, RTGCPTR GCPtrMem)
10117{
10118 /* The lazy approach for now... */
10119 uint8_t const *pbSrc;
10120 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pbSrc, sizeof(*pbSrc), iSegReg, GCPtrMem, IEM_ACCESS_SYS_R);
10121 if (rc == VINF_SUCCESS)
10122 {
10123 *pbDst = *pbSrc;
10124 rc = iemMemCommitAndUnmap(pVCpu, (void *)pbSrc, IEM_ACCESS_SYS_R);
10125 }
10126 return rc;
10127}
10128
10129
10130/**
10131 * Fetches a system table word.
10132 *
10133 * @returns Strict VBox status code.
10134 * @param pVCpu The cross context virtual CPU structure of the calling thread.
10135 * @param pu16Dst Where to return the word.
10136 * @param iSegReg The index of the segment register to use for
10137 * this access. The base and limits are checked.
10138 * @param GCPtrMem The address of the guest memory.
10139 */
10140IEM_STATIC VBOXSTRICTRC iemMemFetchSysU16(PVMCPU pVCpu, uint16_t *pu16Dst, uint8_t iSegReg, RTGCPTR GCPtrMem)
10141{
10142 /* The lazy approach for now... */
10143 uint16_t const *pu16Src;
10144 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu16Src, sizeof(*pu16Src), iSegReg, GCPtrMem, IEM_ACCESS_SYS_R);
10145 if (rc == VINF_SUCCESS)
10146 {
10147 *pu16Dst = *pu16Src;
10148 rc = iemMemCommitAndUnmap(pVCpu, (void *)pu16Src, IEM_ACCESS_SYS_R);
10149 }
10150 return rc;
10151}
10152
10153
10154/**
10155 * Fetches a system table dword.
10156 *
10157 * @returns Strict VBox status code.
10158 * @param pVCpu The cross context virtual CPU structure of the calling thread.
10159 * @param pu32Dst Where to return the dword.
10160 * @param iSegReg The index of the segment register to use for
10161 * this access. The base and limits are checked.
10162 * @param GCPtrMem The address of the guest memory.
10163 */
10164IEM_STATIC VBOXSTRICTRC iemMemFetchSysU32(PVMCPU pVCpu, uint32_t *pu32Dst, uint8_t iSegReg, RTGCPTR GCPtrMem)
10165{
10166 /* The lazy approach for now... */
10167 uint32_t const *pu32Src;
10168 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu32Src, sizeof(*pu32Src), iSegReg, GCPtrMem, IEM_ACCESS_SYS_R);
10169 if (rc == VINF_SUCCESS)
10170 {
10171 *pu32Dst = *pu32Src;
10172 rc = iemMemCommitAndUnmap(pVCpu, (void *)pu32Src, IEM_ACCESS_SYS_R);
10173 }
10174 return rc;
10175}
10176
10177
10178/**
10179 * Fetches a system table qword.
10180 *
10181 * @returns Strict VBox status code.
10182 * @param pVCpu The cross context virtual CPU structure of the calling thread.
10183 * @param pu64Dst Where to return the qword.
10184 * @param iSegReg The index of the segment register to use for
10185 * this access. The base and limits are checked.
10186 * @param GCPtrMem The address of the guest memory.
10187 */
10188IEM_STATIC VBOXSTRICTRC iemMemFetchSysU64(PVMCPU pVCpu, uint64_t *pu64Dst, uint8_t iSegReg, RTGCPTR GCPtrMem)
10189{
10190 /* The lazy approach for now... */
10191 uint64_t const *pu64Src;
10192 VBOXSTRICTRC rc = iemMemMap(pVCpu, (void **)&pu64Src, sizeof(*pu64Src), iSegReg, GCPtrMem, IEM_ACCESS_SYS_R);
10193 if (rc == VINF_SUCCESS)
10194 {
10195 *pu64Dst = *pu64Src;
10196 rc = iemMemCommitAndUnmap(pVCpu, (void *)pu64Src, IEM_ACCESS_SYS_R);
10197 }
10198 return rc;
10199}
10200
10201
10202/**
10203 * Fetches a descriptor table entry with caller specified error code.
10204 *
10205 * @returns Strict VBox status code.
10206 * @param pVCpu The cross context virtual CPU structure of the calling thread.
10207 * @param pDesc Where to return the descriptor table entry.
10208 * @param uSel The selector which table entry to fetch.
10209 * @param uXcpt The exception to raise on table lookup error.
10210 * @param uErrorCode The error code associated with the exception.
10211 */
10212IEM_STATIC VBOXSTRICTRC
10213iemMemFetchSelDescWithErr(PVMCPU pVCpu, PIEMSELDESC pDesc, uint16_t uSel, uint8_t uXcpt, uint16_t uErrorCode)
10214{
10215 AssertPtr(pDesc);
10216 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
10217
10218 /** @todo did the 286 require all 8 bytes to be accessible? */
10219 /*
10220 * Get the selector table base and check bounds.
10221 */
10222 RTGCPTR GCPtrBase;
10223 if (uSel & X86_SEL_LDT)
10224 {
10225 if ( !pCtx->ldtr.Attr.n.u1Present
10226 || (uSel | X86_SEL_RPL_LDT) > pCtx->ldtr.u32Limit )
10227 {
10228 Log(("iemMemFetchSelDesc: LDT selector %#x is out of bounds (%3x) or ldtr is NP (%#x)\n",
10229 uSel, pCtx->ldtr.u32Limit, pCtx->ldtr.Sel));
10230 return iemRaiseXcptOrInt(pVCpu, 0, uXcpt, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR,
10231 uErrorCode, 0);
10232 }
10233
10234 Assert(pCtx->ldtr.Attr.n.u1Present);
10235 GCPtrBase = pCtx->ldtr.u64Base;
10236 }
10237 else
10238 {
10239 if ((uSel | X86_SEL_RPL_LDT) > pCtx->gdtr.cbGdt)
10240 {
10241 Log(("iemMemFetchSelDesc: GDT selector %#x is out of bounds (%3x)\n", uSel, pCtx->gdtr.cbGdt));
10242 return iemRaiseXcptOrInt(pVCpu, 0, uXcpt, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR,
10243 uErrorCode, 0);
10244 }
10245 GCPtrBase = pCtx->gdtr.pGdt;
10246 }
10247
10248 /*
10249 * Read the legacy descriptor and maybe the long mode extensions if
10250 * required.
10251 */
10252 VBOXSTRICTRC rcStrict = iemMemFetchSysU64(pVCpu, &pDesc->Legacy.u, UINT8_MAX, GCPtrBase + (uSel & X86_SEL_MASK));
10253 if (rcStrict == VINF_SUCCESS)
10254 {
10255 if ( !IEM_IS_LONG_MODE(pVCpu)
10256 || pDesc->Legacy.Gen.u1DescType)
10257 pDesc->Long.au64[1] = 0;
10258 else if ((uint32_t)(uSel | X86_SEL_RPL_LDT) + 8 <= (uSel & X86_SEL_LDT ? pCtx->ldtr.u32Limit : pCtx->gdtr.cbGdt))
10259 rcStrict = iemMemFetchSysU64(pVCpu, &pDesc->Long.au64[1], UINT8_MAX, GCPtrBase + (uSel | X86_SEL_RPL_LDT) + 1);
10260 else
10261 {
10262 Log(("iemMemFetchSelDesc: system selector %#x is out of bounds\n", uSel));
10263 /** @todo is this the right exception? */
10264 return iemRaiseXcptOrInt(pVCpu, 0, uXcpt, IEM_XCPT_FLAGS_T_CPU_XCPT | IEM_XCPT_FLAGS_ERR, uErrorCode, 0);
10265 }
10266 }
10267 return rcStrict;
10268}
10269
10270
10271/**
10272 * Fetches a descriptor table entry.
10273 *
10274 * @returns Strict VBox status code.
10275 * @param pVCpu The cross context virtual CPU structure of the calling thread.
10276 * @param pDesc Where to return the descriptor table entry.
10277 * @param uSel The selector which table entry to fetch.
10278 * @param uXcpt The exception to raise on table lookup error.
10279 */
10280IEM_STATIC VBOXSTRICTRC iemMemFetchSelDesc(PVMCPU pVCpu, PIEMSELDESC pDesc, uint16_t uSel, uint8_t uXcpt)
10281{
10282 return iemMemFetchSelDescWithErr(pVCpu, pDesc, uSel, uXcpt, uSel & X86_SEL_MASK_OFF_RPL);
10283}
10284
10285
10286/**
10287 * Fakes a long mode stack selector for SS = 0.
10288 *
10289 * @param pDescSs Where to return the fake stack descriptor.
10290 * @param uDpl The DPL we want.
10291 */
10292IEM_STATIC void iemMemFakeStackSelDesc(PIEMSELDESC pDescSs, uint32_t uDpl)
10293{
10294 pDescSs->Long.au64[0] = 0;
10295 pDescSs->Long.au64[1] = 0;
10296 pDescSs->Long.Gen.u4Type = X86_SEL_TYPE_RW_ACC;
10297 pDescSs->Long.Gen.u1DescType = 1; /* 1 = code / data, 0 = system. */
10298 pDescSs->Long.Gen.u2Dpl = uDpl;
10299 pDescSs->Long.Gen.u1Present = 1;
10300 pDescSs->Long.Gen.u1Long = 1;
10301}
10302
10303
10304/**
10305 * Marks the selector descriptor as accessed (only non-system descriptors).
10306 *
10307 * This function ASSUMES that iemMemFetchSelDesc has be called previously and
10308 * will therefore skip the limit checks.
10309 *
10310 * @returns Strict VBox status code.
10311 * @param pVCpu The cross context virtual CPU structure of the calling thread.
10312 * @param uSel The selector.
10313 */
10314IEM_STATIC VBOXSTRICTRC iemMemMarkSelDescAccessed(PVMCPU pVCpu, uint16_t uSel)
10315{
10316 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
10317
10318 /*
10319 * Get the selector table base and calculate the entry address.
10320 */
10321 RTGCPTR GCPtr = uSel & X86_SEL_LDT
10322 ? pCtx->ldtr.u64Base
10323 : pCtx->gdtr.pGdt;
10324 GCPtr += uSel & X86_SEL_MASK;
10325
10326 /*
10327 * ASMAtomicBitSet will assert if the address is misaligned, so do some
10328 * ugly stuff to avoid this. This will make sure it's an atomic access
10329 * as well more or less remove any question about 8-bit or 32-bit accesss.
10330 */
10331 VBOXSTRICTRC rcStrict;
10332 uint32_t volatile *pu32;
10333 if ((GCPtr & 3) == 0)
10334 {
10335 /* The normal case, map the 32-bit bits around the accessed bit (40). */
10336 GCPtr += 2 + 2;
10337 rcStrict = iemMemMap(pVCpu, (void **)&pu32, 4, UINT8_MAX, GCPtr, IEM_ACCESS_SYS_RW);
10338 if (rcStrict != VINF_SUCCESS)
10339 return rcStrict;
10340 ASMAtomicBitSet(pu32, 8); /* X86_SEL_TYPE_ACCESSED is 1, but it is preceeded by u8BaseHigh1. */
10341 }
10342 else
10343 {
10344 /* The misaligned GDT/LDT case, map the whole thing. */
10345 rcStrict = iemMemMap(pVCpu, (void **)&pu32, 8, UINT8_MAX, GCPtr, IEM_ACCESS_SYS_RW);
10346 if (rcStrict != VINF_SUCCESS)
10347 return rcStrict;
10348 switch ((uintptr_t)pu32 & 3)
10349 {
10350 case 0: ASMAtomicBitSet(pu32, 40 + 0 - 0); break;
10351 case 1: ASMAtomicBitSet((uint8_t volatile *)pu32 + 3, 40 + 0 - 24); break;
10352 case 2: ASMAtomicBitSet((uint8_t volatile *)pu32 + 2, 40 + 0 - 16); break;
10353 case 3: ASMAtomicBitSet((uint8_t volatile *)pu32 + 1, 40 + 0 - 8); break;
10354 }
10355 }
10356
10357 return iemMemCommitAndUnmap(pVCpu, (void *)pu32, IEM_ACCESS_SYS_RW);
10358}
10359
10360/** @} */
10361
10362
10363/*
10364 * Include the C/C++ implementation of instruction.
10365 */
10366#include "IEMAllCImpl.cpp.h"
10367
10368
10369
10370/** @name "Microcode" macros.
10371 *
10372 * The idea is that we should be able to use the same code to interpret
10373 * instructions as well as recompiler instructions. Thus this obfuscation.
10374 *
10375 * @{
10376 */
10377#define IEM_MC_BEGIN(a_cArgs, a_cLocals) {
10378#define IEM_MC_END() }
10379#define IEM_MC_PAUSE() do {} while (0)
10380#define IEM_MC_CONTINUE() do {} while (0)
10381
10382/** Internal macro. */
10383#define IEM_MC_RETURN_ON_FAILURE(a_Expr) \
10384 do \
10385 { \
10386 VBOXSTRICTRC rcStrict2 = a_Expr; \
10387 if (rcStrict2 != VINF_SUCCESS) \
10388 return rcStrict2; \
10389 } while (0)
10390
10391
10392#define IEM_MC_ADVANCE_RIP() iemRegUpdateRipAndClearRF(pVCpu)
10393#define IEM_MC_REL_JMP_S8(a_i8) IEM_MC_RETURN_ON_FAILURE(iemRegRipRelativeJumpS8(pVCpu, a_i8))
10394#define IEM_MC_REL_JMP_S16(a_i16) IEM_MC_RETURN_ON_FAILURE(iemRegRipRelativeJumpS16(pVCpu, a_i16))
10395#define IEM_MC_REL_JMP_S32(a_i32) IEM_MC_RETURN_ON_FAILURE(iemRegRipRelativeJumpS32(pVCpu, a_i32))
10396#define IEM_MC_SET_RIP_U16(a_u16NewIP) IEM_MC_RETURN_ON_FAILURE(iemRegRipJump((pVCpu), (a_u16NewIP)))
10397#define IEM_MC_SET_RIP_U32(a_u32NewIP) IEM_MC_RETURN_ON_FAILURE(iemRegRipJump((pVCpu), (a_u32NewIP)))
10398#define IEM_MC_SET_RIP_U64(a_u64NewIP) IEM_MC_RETURN_ON_FAILURE(iemRegRipJump((pVCpu), (a_u64NewIP)))
10399#define IEM_MC_RAISE_DIVIDE_ERROR() return iemRaiseDivideError(pVCpu)
10400#define IEM_MC_MAYBE_RAISE_DEVICE_NOT_AVAILABLE() \
10401 do { \
10402 if ((pVCpu)->iem.s.CTX_SUFF(pCtx)->cr0 & (X86_CR0_EM | X86_CR0_TS)) \
10403 return iemRaiseDeviceNotAvailable(pVCpu); \
10404 } while (0)
10405#define IEM_MC_MAYBE_RAISE_FPU_XCPT() \
10406 do { \
10407 if ((pVCpu)->iem.s.CTX_SUFF(pCtx)->CTX_SUFF(pXState)->x87.FSW & X86_FSW_ES) \
10408 return iemRaiseMathFault(pVCpu); \
10409 } while (0)
10410#define IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT() \
10411 do { \
10412 if ( (IEM_GET_CTX(pVCpu)->cr0 & X86_CR0_EM) \
10413 || !(IEM_GET_CTX(pVCpu)->cr4 & X86_CR4_OSFXSR) \
10414 || !IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fSse2) \
10415 return iemRaiseUndefinedOpcode(pVCpu); \
10416 if (IEM_GET_CTX(pVCpu)->cr0 & X86_CR0_TS) \
10417 return iemRaiseDeviceNotAvailable(pVCpu); \
10418 } while (0)
10419#define IEM_MC_MAYBE_RAISE_SSE_RELATED_XCPT() \
10420 do { \
10421 if ( (IEM_GET_CTX(pVCpu)->cr0 & X86_CR0_EM) \
10422 || !(IEM_GET_CTX(pVCpu)->cr4 & X86_CR4_OSFXSR) \
10423 || !IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fSse) \
10424 return iemRaiseUndefinedOpcode(pVCpu); \
10425 if (IEM_GET_CTX(pVCpu)->cr0 & X86_CR0_TS) \
10426 return iemRaiseDeviceNotAvailable(pVCpu); \
10427 } while (0)
10428#define IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT() \
10429 do { \
10430 if ( ((pVCpu)->iem.s.CTX_SUFF(pCtx)->cr0 & X86_CR0_EM) \
10431 || !IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fMmx) \
10432 return iemRaiseUndefinedOpcode(pVCpu); \
10433 if (IEM_GET_CTX(pVCpu)->cr0 & X86_CR0_TS) \
10434 return iemRaiseDeviceNotAvailable(pVCpu); \
10435 } while (0)
10436#define IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT_CHECK_SSE_OR_MMXEXT() \
10437 do { \
10438 if ( ((pVCpu)->iem.s.CTX_SUFF(pCtx)->cr0 & X86_CR0_EM) \
10439 || ( !IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fSse \
10440 && !IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fAmdMmxExts) ) \
10441 return iemRaiseUndefinedOpcode(pVCpu); \
10442 if (IEM_GET_CTX(pVCpu)->cr0 & X86_CR0_TS) \
10443 return iemRaiseDeviceNotAvailable(pVCpu); \
10444 } while (0)
10445#define IEM_MC_RAISE_GP0_IF_CPL_NOT_ZERO() \
10446 do { \
10447 if (pVCpu->iem.s.uCpl != 0) \
10448 return iemRaiseGeneralProtectionFault0(pVCpu); \
10449 } while (0)
10450
10451
10452#define IEM_MC_LOCAL(a_Type, a_Name) a_Type a_Name
10453#define IEM_MC_LOCAL_CONST(a_Type, a_Name, a_Value) a_Type const a_Name = (a_Value)
10454#define IEM_MC_REF_LOCAL(a_pRefArg, a_Local) (a_pRefArg) = &(a_Local)
10455#define IEM_MC_ARG(a_Type, a_Name, a_iArg) a_Type a_Name
10456#define IEM_MC_ARG_CONST(a_Type, a_Name, a_Value, a_iArg) a_Type const a_Name = (a_Value)
10457#define IEM_MC_ARG_LOCAL_REF(a_Type, a_Name, a_Local, a_iArg) a_Type const a_Name = &(a_Local)
10458#define IEM_MC_ARG_LOCAL_EFLAGS(a_pName, a_Name, a_iArg) \
10459 uint32_t a_Name; \
10460 uint32_t *a_pName = &a_Name
10461#define IEM_MC_COMMIT_EFLAGS(a_EFlags) \
10462 do { (pVCpu)->iem.s.CTX_SUFF(pCtx)->eflags.u = (a_EFlags); Assert((pVCpu)->iem.s.CTX_SUFF(pCtx)->eflags.u & X86_EFL_1); } while (0)
10463
10464#define IEM_MC_ASSIGN(a_VarOrArg, a_CVariableOrConst) (a_VarOrArg) = (a_CVariableOrConst)
10465#define IEM_MC_ASSIGN_TO_SMALLER IEM_MC_ASSIGN
10466
10467#define IEM_MC_FETCH_GREG_U8(a_u8Dst, a_iGReg) (a_u8Dst) = iemGRegFetchU8(pVCpu, (a_iGReg))
10468#define IEM_MC_FETCH_GREG_U8_ZX_U16(a_u16Dst, a_iGReg) (a_u16Dst) = iemGRegFetchU8(pVCpu, (a_iGReg))
10469#define IEM_MC_FETCH_GREG_U8_ZX_U32(a_u32Dst, a_iGReg) (a_u32Dst) = iemGRegFetchU8(pVCpu, (a_iGReg))
10470#define IEM_MC_FETCH_GREG_U8_ZX_U64(a_u64Dst, a_iGReg) (a_u64Dst) = iemGRegFetchU8(pVCpu, (a_iGReg))
10471#define IEM_MC_FETCH_GREG_U8_SX_U16(a_u16Dst, a_iGReg) (a_u16Dst) = (int8_t)iemGRegFetchU8(pVCpu, (a_iGReg))
10472#define IEM_MC_FETCH_GREG_U8_SX_U32(a_u32Dst, a_iGReg) (a_u32Dst) = (int8_t)iemGRegFetchU8(pVCpu, (a_iGReg))
10473#define IEM_MC_FETCH_GREG_U8_SX_U64(a_u64Dst, a_iGReg) (a_u64Dst) = (int8_t)iemGRegFetchU8(pVCpu, (a_iGReg))
10474#define IEM_MC_FETCH_GREG_U16(a_u16Dst, a_iGReg) (a_u16Dst) = iemGRegFetchU16(pVCpu, (a_iGReg))
10475#define IEM_MC_FETCH_GREG_U16_ZX_U32(a_u32Dst, a_iGReg) (a_u32Dst) = iemGRegFetchU16(pVCpu, (a_iGReg))
10476#define IEM_MC_FETCH_GREG_U16_ZX_U64(a_u64Dst, a_iGReg) (a_u64Dst) = iemGRegFetchU16(pVCpu, (a_iGReg))
10477#define IEM_MC_FETCH_GREG_U16_SX_U32(a_u32Dst, a_iGReg) (a_u32Dst) = (int16_t)iemGRegFetchU16(pVCpu, (a_iGReg))
10478#define IEM_MC_FETCH_GREG_U16_SX_U64(a_u64Dst, a_iGReg) (a_u64Dst) = (int16_t)iemGRegFetchU16(pVCpu, (a_iGReg))
10479#define IEM_MC_FETCH_GREG_U32(a_u32Dst, a_iGReg) (a_u32Dst) = iemGRegFetchU32(pVCpu, (a_iGReg))
10480#define IEM_MC_FETCH_GREG_U32_ZX_U64(a_u64Dst, a_iGReg) (a_u64Dst) = iemGRegFetchU32(pVCpu, (a_iGReg))
10481#define IEM_MC_FETCH_GREG_U32_SX_U64(a_u64Dst, a_iGReg) (a_u64Dst) = (int32_t)iemGRegFetchU32(pVCpu, (a_iGReg))
10482#define IEM_MC_FETCH_GREG_U64(a_u64Dst, a_iGReg) (a_u64Dst) = iemGRegFetchU64(pVCpu, (a_iGReg))
10483#define IEM_MC_FETCH_GREG_U64_ZX_U64 IEM_MC_FETCH_GREG_U64
10484#define IEM_MC_FETCH_SREG_U16(a_u16Dst, a_iSReg) (a_u16Dst) = iemSRegFetchU16(pVCpu, (a_iSReg))
10485#define IEM_MC_FETCH_SREG_ZX_U32(a_u32Dst, a_iSReg) (a_u32Dst) = iemSRegFetchU16(pVCpu, (a_iSReg))
10486#define IEM_MC_FETCH_SREG_ZX_U64(a_u64Dst, a_iSReg) (a_u64Dst) = iemSRegFetchU16(pVCpu, (a_iSReg))
10487#define IEM_MC_FETCH_CR0_U16(a_u16Dst) (a_u16Dst) = (uint16_t)(pVCpu)->iem.s.CTX_SUFF(pCtx)->cr0
10488#define IEM_MC_FETCH_CR0_U32(a_u32Dst) (a_u32Dst) = (uint32_t)(pVCpu)->iem.s.CTX_SUFF(pCtx)->cr0
10489#define IEM_MC_FETCH_CR0_U64(a_u64Dst) (a_u64Dst) = (pVCpu)->iem.s.CTX_SUFF(pCtx)->cr0
10490#define IEM_MC_FETCH_LDTR_U16(a_u16Dst) (a_u16Dst) = (pVCpu)->iem.s.CTX_SUFF(pCtx)->ldtr.Sel
10491#define IEM_MC_FETCH_LDTR_U32(a_u32Dst) (a_u32Dst) = (pVCpu)->iem.s.CTX_SUFF(pCtx)->ldtr.Sel
10492#define IEM_MC_FETCH_LDTR_U64(a_u64Dst) (a_u64Dst) = (pVCpu)->iem.s.CTX_SUFF(pCtx)->ldtr.Sel
10493#define IEM_MC_FETCH_TR_U16(a_u16Dst) (a_u16Dst) = (pVCpu)->iem.s.CTX_SUFF(pCtx)->tr.Sel
10494#define IEM_MC_FETCH_TR_U32(a_u32Dst) (a_u32Dst) = (pVCpu)->iem.s.CTX_SUFF(pCtx)->tr.Sel
10495#define IEM_MC_FETCH_TR_U64(a_u64Dst) (a_u64Dst) = (pVCpu)->iem.s.CTX_SUFF(pCtx)->tr.Sel
10496/** @note Not for IOPL or IF testing or modification. */
10497#define IEM_MC_FETCH_EFLAGS(a_EFlags) (a_EFlags) = (pVCpu)->iem.s.CTX_SUFF(pCtx)->eflags.u
10498#define IEM_MC_FETCH_EFLAGS_U8(a_EFlags) (a_EFlags) = (uint8_t)(pVCpu)->iem.s.CTX_SUFF(pCtx)->eflags.u
10499#define IEM_MC_FETCH_FSW(a_u16Fsw) (a_u16Fsw) = IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.FSW
10500#define IEM_MC_FETCH_FCW(a_u16Fcw) (a_u16Fcw) = IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.FCW
10501
10502#define IEM_MC_STORE_GREG_U8(a_iGReg, a_u8Value) *iemGRegRefU8( pVCpu, (a_iGReg)) = (a_u8Value)
10503#define IEM_MC_STORE_GREG_U16(a_iGReg, a_u16Value) *iemGRegRefU16(pVCpu, (a_iGReg)) = (a_u16Value)
10504#define IEM_MC_STORE_GREG_U32(a_iGReg, a_u32Value) *iemGRegRefU64(pVCpu, (a_iGReg)) = (uint32_t)(a_u32Value) /* clear high bits. */
10505#define IEM_MC_STORE_GREG_U64(a_iGReg, a_u64Value) *iemGRegRefU64(pVCpu, (a_iGReg)) = (a_u64Value)
10506#define IEM_MC_STORE_GREG_U8_CONST IEM_MC_STORE_GREG_U8
10507#define IEM_MC_STORE_GREG_U16_CONST IEM_MC_STORE_GREG_U16
10508#define IEM_MC_STORE_GREG_U32_CONST IEM_MC_STORE_GREG_U32
10509#define IEM_MC_STORE_GREG_U64_CONST IEM_MC_STORE_GREG_U64
10510#define IEM_MC_CLEAR_HIGH_GREG_U64(a_iGReg) *iemGRegRefU64(pVCpu, (a_iGReg)) &= UINT32_MAX
10511#define IEM_MC_CLEAR_HIGH_GREG_U64_BY_REF(a_pu32Dst) do { (a_pu32Dst)[1] = 0; } while (0)
10512#define IEM_MC_STORE_FPUREG_R80_SRC_REF(a_iSt, a_pr80Src) \
10513 do { IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aRegs[a_iSt].r80 = *(a_pr80Src); } while (0)
10514
10515#define IEM_MC_REF_GREG_U8(a_pu8Dst, a_iGReg) (a_pu8Dst) = iemGRegRefU8( pVCpu, (a_iGReg))
10516#define IEM_MC_REF_GREG_U16(a_pu16Dst, a_iGReg) (a_pu16Dst) = iemGRegRefU16(pVCpu, (a_iGReg))
10517/** @todo User of IEM_MC_REF_GREG_U32 needs to clear the high bits on commit.
10518 * Use IEM_MC_CLEAR_HIGH_GREG_U64_BY_REF! */
10519#define IEM_MC_REF_GREG_U32(a_pu32Dst, a_iGReg) (a_pu32Dst) = iemGRegRefU32(pVCpu, (a_iGReg))
10520#define IEM_MC_REF_GREG_U64(a_pu64Dst, a_iGReg) (a_pu64Dst) = iemGRegRefU64(pVCpu, (a_iGReg))
10521/** @note Not for IOPL or IF testing or modification. */
10522#define IEM_MC_REF_EFLAGS(a_pEFlags) (a_pEFlags) = &(pVCpu)->iem.s.CTX_SUFF(pCtx)->eflags.u
10523
10524#define IEM_MC_ADD_GREG_U8(a_iGReg, a_u8Value) *iemGRegRefU8( pVCpu, (a_iGReg)) += (a_u8Value)
10525#define IEM_MC_ADD_GREG_U16(a_iGReg, a_u16Value) *iemGRegRefU16(pVCpu, (a_iGReg)) += (a_u16Value)
10526#define IEM_MC_ADD_GREG_U32(a_iGReg, a_u32Value) \
10527 do { \
10528 uint32_t *pu32Reg = iemGRegRefU32(pVCpu, (a_iGReg)); \
10529 *pu32Reg += (a_u32Value); \
10530 pu32Reg[1] = 0; /* implicitly clear the high bit. */ \
10531 } while (0)
10532#define IEM_MC_ADD_GREG_U64(a_iGReg, a_u64Value) *iemGRegRefU64(pVCpu, (a_iGReg)) += (a_u64Value)
10533
10534#define IEM_MC_SUB_GREG_U8(a_iGReg, a_u8Value) *iemGRegRefU8( pVCpu, (a_iGReg)) -= (a_u8Value)
10535#define IEM_MC_SUB_GREG_U16(a_iGReg, a_u16Value) *iemGRegRefU16(pVCpu, (a_iGReg)) -= (a_u16Value)
10536#define IEM_MC_SUB_GREG_U32(a_iGReg, a_u32Value) \
10537 do { \
10538 uint32_t *pu32Reg = iemGRegRefU32(pVCpu, (a_iGReg)); \
10539 *pu32Reg -= (a_u32Value); \
10540 pu32Reg[1] = 0; /* implicitly clear the high bit. */ \
10541 } while (0)
10542#define IEM_MC_SUB_GREG_U64(a_iGReg, a_u64Value) *iemGRegRefU64(pVCpu, (a_iGReg)) -= (a_u64Value)
10543#define IEM_MC_SUB_LOCAL_U16(a_u16Value, a_u16Const) do { (a_u16Value) -= a_u16Const; } while (0)
10544
10545#define IEM_MC_ADD_GREG_U8_TO_LOCAL(a_u8Value, a_iGReg) do { (a_u8Value) += iemGRegFetchU8( pVCpu, (a_iGReg)); } while (0)
10546#define IEM_MC_ADD_GREG_U16_TO_LOCAL(a_u16Value, a_iGReg) do { (a_u16Value) += iemGRegFetchU16(pVCpu, (a_iGReg)); } while (0)
10547#define IEM_MC_ADD_GREG_U32_TO_LOCAL(a_u32Value, a_iGReg) do { (a_u32Value) += iemGRegFetchU32(pVCpu, (a_iGReg)); } while (0)
10548#define IEM_MC_ADD_GREG_U64_TO_LOCAL(a_u64Value, a_iGReg) do { (a_u64Value) += iemGRegFetchU64(pVCpu, (a_iGReg)); } while (0)
10549#define IEM_MC_ADD_LOCAL_S16_TO_EFF_ADDR(a_EffAddr, a_i16) do { (a_EffAddr) += (a_i16); } while (0)
10550#define IEM_MC_ADD_LOCAL_S32_TO_EFF_ADDR(a_EffAddr, a_i32) do { (a_EffAddr) += (a_i32); } while (0)
10551#define IEM_MC_ADD_LOCAL_S64_TO_EFF_ADDR(a_EffAddr, a_i64) do { (a_EffAddr) += (a_i64); } while (0)
10552
10553#define IEM_MC_AND_LOCAL_U8(a_u8Local, a_u8Mask) do { (a_u8Local) &= (a_u8Mask); } while (0)
10554#define IEM_MC_AND_LOCAL_U16(a_u16Local, a_u16Mask) do { (a_u16Local) &= (a_u16Mask); } while (0)
10555#define IEM_MC_AND_LOCAL_U32(a_u32Local, a_u32Mask) do { (a_u32Local) &= (a_u32Mask); } while (0)
10556#define IEM_MC_AND_LOCAL_U64(a_u64Local, a_u64Mask) do { (a_u64Local) &= (a_u64Mask); } while (0)
10557
10558#define IEM_MC_AND_ARG_U16(a_u16Arg, a_u16Mask) do { (a_u16Arg) &= (a_u16Mask); } while (0)
10559#define IEM_MC_AND_ARG_U32(a_u32Arg, a_u32Mask) do { (a_u32Arg) &= (a_u32Mask); } while (0)
10560#define IEM_MC_AND_ARG_U64(a_u64Arg, a_u64Mask) do { (a_u64Arg) &= (a_u64Mask); } while (0)
10561
10562#define IEM_MC_OR_LOCAL_U8(a_u8Local, a_u8Mask) do { (a_u8Local) |= (a_u8Mask); } while (0)
10563#define IEM_MC_OR_LOCAL_U16(a_u16Local, a_u16Mask) do { (a_u16Local) |= (a_u16Mask); } while (0)
10564#define IEM_MC_OR_LOCAL_U32(a_u32Local, a_u32Mask) do { (a_u32Local) |= (a_u32Mask); } while (0)
10565
10566#define IEM_MC_SAR_LOCAL_S16(a_i16Local, a_cShift) do { (a_i16Local) >>= (a_cShift); } while (0)
10567#define IEM_MC_SAR_LOCAL_S32(a_i32Local, a_cShift) do { (a_i32Local) >>= (a_cShift); } while (0)
10568#define IEM_MC_SAR_LOCAL_S64(a_i64Local, a_cShift) do { (a_i64Local) >>= (a_cShift); } while (0)
10569
10570#define IEM_MC_SHL_LOCAL_S16(a_i16Local, a_cShift) do { (a_i16Local) <<= (a_cShift); } while (0)
10571#define IEM_MC_SHL_LOCAL_S32(a_i32Local, a_cShift) do { (a_i32Local) <<= (a_cShift); } while (0)
10572#define IEM_MC_SHL_LOCAL_S64(a_i64Local, a_cShift) do { (a_i64Local) <<= (a_cShift); } while (0)
10573
10574#define IEM_MC_AND_2LOCS_U32(a_u32Local, a_u32Mask) do { (a_u32Local) &= (a_u32Mask); } while (0)
10575
10576#define IEM_MC_OR_2LOCS_U32(a_u32Local, a_u32Mask) do { (a_u32Local) |= (a_u32Mask); } while (0)
10577
10578#define IEM_MC_AND_GREG_U8(a_iGReg, a_u8Value) *iemGRegRefU8( pVCpu, (a_iGReg)) &= (a_u8Value)
10579#define IEM_MC_AND_GREG_U16(a_iGReg, a_u16Value) *iemGRegRefU16(pVCpu, (a_iGReg)) &= (a_u16Value)
10580#define IEM_MC_AND_GREG_U32(a_iGReg, a_u32Value) \
10581 do { \
10582 uint32_t *pu32Reg = iemGRegRefU32(pVCpu, (a_iGReg)); \
10583 *pu32Reg &= (a_u32Value); \
10584 pu32Reg[1] = 0; /* implicitly clear the high bit. */ \
10585 } while (0)
10586#define IEM_MC_AND_GREG_U64(a_iGReg, a_u64Value) *iemGRegRefU64(pVCpu, (a_iGReg)) &= (a_u64Value)
10587
10588#define IEM_MC_OR_GREG_U8(a_iGReg, a_u8Value) *iemGRegRefU8( pVCpu, (a_iGReg)) |= (a_u8Value)
10589#define IEM_MC_OR_GREG_U16(a_iGReg, a_u16Value) *iemGRegRefU16(pVCpu, (a_iGReg)) |= (a_u16Value)
10590#define IEM_MC_OR_GREG_U32(a_iGReg, a_u32Value) \
10591 do { \
10592 uint32_t *pu32Reg = iemGRegRefU32(pVCpu, (a_iGReg)); \
10593 *pu32Reg |= (a_u32Value); \
10594 pu32Reg[1] = 0; /* implicitly clear the high bit. */ \
10595 } while (0)
10596#define IEM_MC_OR_GREG_U64(a_iGReg, a_u64Value) *iemGRegRefU64(pVCpu, (a_iGReg)) |= (a_u64Value)
10597
10598
10599/** @note Not for IOPL or IF modification. */
10600#define IEM_MC_SET_EFL_BIT(a_fBit) do { (pVCpu)->iem.s.CTX_SUFF(pCtx)->eflags.u |= (a_fBit); } while (0)
10601/** @note Not for IOPL or IF modification. */
10602#define IEM_MC_CLEAR_EFL_BIT(a_fBit) do { (pVCpu)->iem.s.CTX_SUFF(pCtx)->eflags.u &= ~(a_fBit); } while (0)
10603/** @note Not for IOPL or IF modification. */
10604#define IEM_MC_FLIP_EFL_BIT(a_fBit) do { (pVCpu)->iem.s.CTX_SUFF(pCtx)->eflags.u ^= (a_fBit); } while (0)
10605
10606#define IEM_MC_CLEAR_FSW_EX() do { (pVCpu)->iem.s.CTX_SUFF(pCtx)->CTX_SUFF(pXState)->x87.FSW &= X86_FSW_C_MASK | X86_FSW_TOP_MASK; } while (0)
10607
10608
10609#define IEM_MC_FETCH_MREG_U64(a_u64Value, a_iMReg) \
10610 do { (a_u64Value) = IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aRegs[(a_iMReg)].mmx; } while (0)
10611#define IEM_MC_FETCH_MREG_U32(a_u32Value, a_iMReg) \
10612 do { (a_u32Value) = IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aRegs[(a_iMReg)].au32[0]; } while (0)
10613#define IEM_MC_STORE_MREG_U64(a_iMReg, a_u64Value) \
10614 do { IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aRegs[(a_iMReg)].mmx = (a_u64Value); } while (0)
10615#define IEM_MC_STORE_MREG_U32_ZX_U64(a_iMReg, a_u32Value) \
10616 do { IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aRegs[(a_iMReg)].mmx = (uint32_t)(a_u32Value); } while (0)
10617#define IEM_MC_REF_MREG_U64(a_pu64Dst, a_iMReg) \
10618 (a_pu64Dst) = (&IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aRegs[(a_iMReg)].mmx)
10619#define IEM_MC_REF_MREG_U64_CONST(a_pu64Dst, a_iMReg) \
10620 (a_pu64Dst) = ((uint64_t const *)&IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aRegs[(a_iMReg)].mmx)
10621#define IEM_MC_REF_MREG_U32_CONST(a_pu32Dst, a_iMReg) \
10622 (a_pu32Dst) = ((uint32_t const *)&IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aRegs[(a_iMReg)].mmx)
10623
10624#define IEM_MC_FETCH_XREG_U128(a_u128Value, a_iXReg) \
10625 do { (a_u128Value) = IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aXMM[(a_iXReg)].xmm; } while (0)
10626#define IEM_MC_FETCH_XREG_U64(a_u64Value, a_iXReg) \
10627 do { (a_u64Value) = IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aXMM[(a_iXReg)].au64[0]; } while (0)
10628#define IEM_MC_FETCH_XREG_U32(a_u32Value, a_iXReg) \
10629 do { (a_u32Value) = IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aXMM[(a_iXReg)].au32[0]; } while (0)
10630#define IEM_MC_STORE_XREG_U128(a_iXReg, a_u128Value) \
10631 do { IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aXMM[(a_iXReg)].xmm = (a_u128Value); } while (0)
10632#define IEM_MC_STORE_XREG_U64(a_iXReg, a_u64Value) \
10633 do { IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aXMM[(a_iXReg)].au64[0] = (a_u64Value); } while (0)
10634#define IEM_MC_STORE_XREG_U64_ZX_U128(a_iXReg, a_u64Value) \
10635 do { IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aXMM[(a_iXReg)].au64[0] = (a_u64Value); \
10636 IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aXMM[(a_iXReg)].au64[1] = 0; \
10637 } while (0)
10638#define IEM_MC_STORE_XREG_U32_ZX_U128(a_iXReg, a_u32Value) \
10639 do { IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aXMM[(a_iXReg)].au64[0] = (uint32_t)(a_u32Value); \
10640 IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aXMM[(a_iXReg)].au64[1] = 0; \
10641 } while (0)
10642#define IEM_MC_REF_XREG_U128(a_pu128Dst, a_iXReg) \
10643 (a_pu128Dst) = (&IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aXMM[(a_iXReg)].xmm)
10644#define IEM_MC_REF_XREG_U128_CONST(a_pu128Dst, a_iXReg) \
10645 (a_pu128Dst) = ((uint128_t const *)&IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aXMM[(a_iXReg)].xmm)
10646#define IEM_MC_REF_XREG_U64_CONST(a_pu64Dst, a_iXReg) \
10647 (a_pu64Dst) = ((uint64_t const *)&IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aXMM[(a_iXReg)].au64[0])
10648#define IEM_MC_COPY_XREG_U128(a_iXRegDst, a_iXRegSrc) \
10649 do { IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aXMM[(a_iXRegDst)].xmm \
10650 = IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.aXMM[(a_iXRegSrc)].xmm; } while (0)
10651
10652#ifndef IEM_WITH_SETJMP
10653# define IEM_MC_FETCH_MEM_U8(a_u8Dst, a_iSeg, a_GCPtrMem) \
10654 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU8(pVCpu, &(a_u8Dst), (a_iSeg), (a_GCPtrMem)))
10655# define IEM_MC_FETCH_MEM16_U8(a_u8Dst, a_iSeg, a_GCPtrMem16) \
10656 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU8(pVCpu, &(a_u8Dst), (a_iSeg), (a_GCPtrMem16)))
10657# define IEM_MC_FETCH_MEM32_U8(a_u8Dst, a_iSeg, a_GCPtrMem32) \
10658 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU8(pVCpu, &(a_u8Dst), (a_iSeg), (a_GCPtrMem32)))
10659#else
10660# define IEM_MC_FETCH_MEM_U8(a_u8Dst, a_iSeg, a_GCPtrMem) \
10661 ((a_u8Dst) = iemMemFetchDataU8Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10662# define IEM_MC_FETCH_MEM16_U8(a_u8Dst, a_iSeg, a_GCPtrMem16) \
10663 ((a_u8Dst) = iemMemFetchDataU8Jmp(pVCpu, (a_iSeg), (a_GCPtrMem16)))
10664# define IEM_MC_FETCH_MEM32_U8(a_u8Dst, a_iSeg, a_GCPtrMem32) \
10665 ((a_u8Dst) = iemMemFetchDataU8Jmp(pVCpu, (a_iSeg), (a_GCPtrMem32)))
10666#endif
10667
10668#ifndef IEM_WITH_SETJMP
10669# define IEM_MC_FETCH_MEM_U16(a_u16Dst, a_iSeg, a_GCPtrMem) \
10670 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU16(pVCpu, &(a_u16Dst), (a_iSeg), (a_GCPtrMem)))
10671# define IEM_MC_FETCH_MEM_U16_DISP(a_u16Dst, a_iSeg, a_GCPtrMem, a_offDisp) \
10672 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU16(pVCpu, &(a_u16Dst), (a_iSeg), (a_GCPtrMem) + (a_offDisp)))
10673# define IEM_MC_FETCH_MEM_I16(a_i16Dst, a_iSeg, a_GCPtrMem) \
10674 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU16(pVCpu, (uint16_t *)&(a_i16Dst), (a_iSeg), (a_GCPtrMem)))
10675#else
10676# define IEM_MC_FETCH_MEM_U16(a_u16Dst, a_iSeg, a_GCPtrMem) \
10677 ((a_u16Dst) = iemMemFetchDataU16Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10678# define IEM_MC_FETCH_MEM_U16_DISP(a_u16Dst, a_iSeg, a_GCPtrMem, a_offDisp) \
10679 ((a_u16Dst) = iemMemFetchDataU16Jmp(pVCpu, (a_iSeg), (a_GCPtrMem) + (a_offDisp)))
10680# define IEM_MC_FETCH_MEM_I16(a_i16Dst, a_iSeg, a_GCPtrMem) \
10681 ((a_i16Dst) = (int16_t)iemMemFetchDataU16Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10682#endif
10683
10684#ifndef IEM_WITH_SETJMP
10685# define IEM_MC_FETCH_MEM_U32(a_u32Dst, a_iSeg, a_GCPtrMem) \
10686 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU32(pVCpu, &(a_u32Dst), (a_iSeg), (a_GCPtrMem)))
10687# define IEM_MC_FETCH_MEM_U32_DISP(a_u32Dst, a_iSeg, a_GCPtrMem, a_offDisp) \
10688 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU32(pVCpu, &(a_u32Dst), (a_iSeg), (a_GCPtrMem) + (a_offDisp)))
10689# define IEM_MC_FETCH_MEM_I32(a_i32Dst, a_iSeg, a_GCPtrMem) \
10690 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU32(pVCpu, (uint32_t *)&(a_i32Dst), (a_iSeg), (a_GCPtrMem)))
10691#else
10692# define IEM_MC_FETCH_MEM_U32(a_u32Dst, a_iSeg, a_GCPtrMem) \
10693 ((a_u32Dst) = iemMemFetchDataU32Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10694# define IEM_MC_FETCH_MEM_U32_DISP(a_u32Dst, a_iSeg, a_GCPtrMem, a_offDisp) \
10695 ((a_u32Dst) = iemMemFetchDataU32Jmp(pVCpu, (a_iSeg), (a_GCPtrMem) + (a_offDisp)))
10696# define IEM_MC_FETCH_MEM_I32(a_i32Dst, a_iSeg, a_GCPtrMem) \
10697 ((a_i32Dst) = (int32_t)iemMemFetchDataU32Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10698#endif
10699
10700#ifdef SOME_UNUSED_FUNCTION
10701# define IEM_MC_FETCH_MEM_S32_SX_U64(a_u64Dst, a_iSeg, a_GCPtrMem) \
10702 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataS32SxU64(pVCpu, &(a_u64Dst), (a_iSeg), (a_GCPtrMem)))
10703#endif
10704
10705#ifndef IEM_WITH_SETJMP
10706# define IEM_MC_FETCH_MEM_U64(a_u64Dst, a_iSeg, a_GCPtrMem) \
10707 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU64(pVCpu, &(a_u64Dst), (a_iSeg), (a_GCPtrMem)))
10708# define IEM_MC_FETCH_MEM_U64_DISP(a_u64Dst, a_iSeg, a_GCPtrMem, a_offDisp) \
10709 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU64(pVCpu, &(a_u64Dst), (a_iSeg), (a_GCPtrMem) + (a_offDisp)))
10710# define IEM_MC_FETCH_MEM_U64_ALIGN_U128(a_u64Dst, a_iSeg, a_GCPtrMem) \
10711 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU64AlignedU128(pVCpu, &(a_u64Dst), (a_iSeg), (a_GCPtrMem)))
10712# define IEM_MC_FETCH_MEM_I64(a_i64Dst, a_iSeg, a_GCPtrMem) \
10713 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU64(pVCpu, (uint64_t *)&(a_i64Dst), (a_iSeg), (a_GCPtrMem)))
10714#else
10715# define IEM_MC_FETCH_MEM_U64(a_u64Dst, a_iSeg, a_GCPtrMem) \
10716 ((a_u64Dst) = iemMemFetchDataU64Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10717# define IEM_MC_FETCH_MEM_U64_DISP(a_u64Dst, a_iSeg, a_GCPtrMem, a_offDisp) \
10718 ((a_u64Dst) = iemMemFetchDataU64Jmp(pVCpu, (a_iSeg), (a_GCPtrMem) + (a_offDisp)))
10719# define IEM_MC_FETCH_MEM_U64_ALIGN_U128(a_u64Dst, a_iSeg, a_GCPtrMem) \
10720 ((a_u64Dst) = iemMemFetchDataU64AlignedU128Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10721# define IEM_MC_FETCH_MEM_I64(a_i64Dst, a_iSeg, a_GCPtrMem) \
10722 ((a_i64Dst) = (int64_t)iemMemFetchDataU64Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10723#endif
10724
10725#ifndef IEM_WITH_SETJMP
10726# define IEM_MC_FETCH_MEM_R32(a_r32Dst, a_iSeg, a_GCPtrMem) \
10727 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU32(pVCpu, &(a_r32Dst).u32, (a_iSeg), (a_GCPtrMem)))
10728# define IEM_MC_FETCH_MEM_R64(a_r64Dst, a_iSeg, a_GCPtrMem) \
10729 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU64(pVCpu, &(a_r64Dst).au64[0], (a_iSeg), (a_GCPtrMem)))
10730# define IEM_MC_FETCH_MEM_R80(a_r80Dst, a_iSeg, a_GCPtrMem) \
10731 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataR80(pVCpu, &(a_r80Dst), (a_iSeg), (a_GCPtrMem)))
10732#else
10733# define IEM_MC_FETCH_MEM_R32(a_r32Dst, a_iSeg, a_GCPtrMem) \
10734 ((a_r32Dst).u32 = iemMemFetchDataU32Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10735# define IEM_MC_FETCH_MEM_R64(a_r64Dst, a_iSeg, a_GCPtrMem) \
10736 ((a_r64Dst).au64[0] = iemMemFetchDataU64Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10737# define IEM_MC_FETCH_MEM_R80(a_r80Dst, a_iSeg, a_GCPtrMem) \
10738 iemMemFetchDataR80Jmp(pVCpu, &(a_r80Dst), (a_iSeg), (a_GCPtrMem))
10739#endif
10740
10741#ifndef IEM_WITH_SETJMP
10742# define IEM_MC_FETCH_MEM_U128(a_u128Dst, a_iSeg, a_GCPtrMem) \
10743 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU128(pVCpu, &(a_u128Dst), (a_iSeg), (a_GCPtrMem)))
10744# define IEM_MC_FETCH_MEM_U128_ALIGN_SSE(a_u128Dst, a_iSeg, a_GCPtrMem) \
10745 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU128AlignedSse(pVCpu, &(a_u128Dst), (a_iSeg), (a_GCPtrMem)))
10746#else
10747# define IEM_MC_FETCH_MEM_U128(a_u128Dst, a_iSeg, a_GCPtrMem) \
10748 iemMemFetchDataU128Jmp(pVCpu, &(a_u128Dst), (a_iSeg), (a_GCPtrMem))
10749# define IEM_MC_FETCH_MEM_U128_ALIGN_SSE(a_u128Dst, a_iSeg, a_GCPtrMem) \
10750 iemMemFetchDataU128AlignedSseJmp(pVCpu, &(a_u128Dst), (a_iSeg), (a_GCPtrMem))
10751#endif
10752
10753
10754
10755#ifndef IEM_WITH_SETJMP
10756# define IEM_MC_FETCH_MEM_U8_ZX_U16(a_u16Dst, a_iSeg, a_GCPtrMem) \
10757 do { \
10758 uint8_t u8Tmp; \
10759 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU8(pVCpu, &u8Tmp, (a_iSeg), (a_GCPtrMem))); \
10760 (a_u16Dst) = u8Tmp; \
10761 } while (0)
10762# define IEM_MC_FETCH_MEM_U8_ZX_U32(a_u32Dst, a_iSeg, a_GCPtrMem) \
10763 do { \
10764 uint8_t u8Tmp; \
10765 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU8(pVCpu, &u8Tmp, (a_iSeg), (a_GCPtrMem))); \
10766 (a_u32Dst) = u8Tmp; \
10767 } while (0)
10768# define IEM_MC_FETCH_MEM_U8_ZX_U64(a_u64Dst, a_iSeg, a_GCPtrMem) \
10769 do { \
10770 uint8_t u8Tmp; \
10771 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU8(pVCpu, &u8Tmp, (a_iSeg), (a_GCPtrMem))); \
10772 (a_u64Dst) = u8Tmp; \
10773 } while (0)
10774# define IEM_MC_FETCH_MEM_U16_ZX_U32(a_u32Dst, a_iSeg, a_GCPtrMem) \
10775 do { \
10776 uint16_t u16Tmp; \
10777 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU16(pVCpu, &u16Tmp, (a_iSeg), (a_GCPtrMem))); \
10778 (a_u32Dst) = u16Tmp; \
10779 } while (0)
10780# define IEM_MC_FETCH_MEM_U16_ZX_U64(a_u64Dst, a_iSeg, a_GCPtrMem) \
10781 do { \
10782 uint16_t u16Tmp; \
10783 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU16(pVCpu, &u16Tmp, (a_iSeg), (a_GCPtrMem))); \
10784 (a_u64Dst) = u16Tmp; \
10785 } while (0)
10786# define IEM_MC_FETCH_MEM_U32_ZX_U64(a_u64Dst, a_iSeg, a_GCPtrMem) \
10787 do { \
10788 uint32_t u32Tmp; \
10789 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU32(pVCpu, &u32Tmp, (a_iSeg), (a_GCPtrMem))); \
10790 (a_u64Dst) = u32Tmp; \
10791 } while (0)
10792#else /* IEM_WITH_SETJMP */
10793# define IEM_MC_FETCH_MEM_U8_ZX_U16(a_u16Dst, a_iSeg, a_GCPtrMem) \
10794 ((a_u16Dst) = iemMemFetchDataU8Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10795# define IEM_MC_FETCH_MEM_U8_ZX_U32(a_u32Dst, a_iSeg, a_GCPtrMem) \
10796 ((a_u32Dst) = iemMemFetchDataU8Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10797# define IEM_MC_FETCH_MEM_U8_ZX_U64(a_u64Dst, a_iSeg, a_GCPtrMem) \
10798 ((a_u64Dst) = iemMemFetchDataU8Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10799# define IEM_MC_FETCH_MEM_U16_ZX_U32(a_u32Dst, a_iSeg, a_GCPtrMem) \
10800 ((a_u32Dst) = iemMemFetchDataU16Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10801# define IEM_MC_FETCH_MEM_U16_ZX_U64(a_u64Dst, a_iSeg, a_GCPtrMem) \
10802 ((a_u64Dst) = iemMemFetchDataU16Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10803# define IEM_MC_FETCH_MEM_U32_ZX_U64(a_u64Dst, a_iSeg, a_GCPtrMem) \
10804 ((a_u64Dst) = iemMemFetchDataU32Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10805#endif /* IEM_WITH_SETJMP */
10806
10807#ifndef IEM_WITH_SETJMP
10808# define IEM_MC_FETCH_MEM_U8_SX_U16(a_u16Dst, a_iSeg, a_GCPtrMem) \
10809 do { \
10810 uint8_t u8Tmp; \
10811 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU8(pVCpu, &u8Tmp, (a_iSeg), (a_GCPtrMem))); \
10812 (a_u16Dst) = (int8_t)u8Tmp; \
10813 } while (0)
10814# define IEM_MC_FETCH_MEM_U8_SX_U32(a_u32Dst, a_iSeg, a_GCPtrMem) \
10815 do { \
10816 uint8_t u8Tmp; \
10817 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU8(pVCpu, &u8Tmp, (a_iSeg), (a_GCPtrMem))); \
10818 (a_u32Dst) = (int8_t)u8Tmp; \
10819 } while (0)
10820# define IEM_MC_FETCH_MEM_U8_SX_U64(a_u64Dst, a_iSeg, a_GCPtrMem) \
10821 do { \
10822 uint8_t u8Tmp; \
10823 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU8(pVCpu, &u8Tmp, (a_iSeg), (a_GCPtrMem))); \
10824 (a_u64Dst) = (int8_t)u8Tmp; \
10825 } while (0)
10826# define IEM_MC_FETCH_MEM_U16_SX_U32(a_u32Dst, a_iSeg, a_GCPtrMem) \
10827 do { \
10828 uint16_t u16Tmp; \
10829 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU16(pVCpu, &u16Tmp, (a_iSeg), (a_GCPtrMem))); \
10830 (a_u32Dst) = (int16_t)u16Tmp; \
10831 } while (0)
10832# define IEM_MC_FETCH_MEM_U16_SX_U64(a_u64Dst, a_iSeg, a_GCPtrMem) \
10833 do { \
10834 uint16_t u16Tmp; \
10835 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU16(pVCpu, &u16Tmp, (a_iSeg), (a_GCPtrMem))); \
10836 (a_u64Dst) = (int16_t)u16Tmp; \
10837 } while (0)
10838# define IEM_MC_FETCH_MEM_U32_SX_U64(a_u64Dst, a_iSeg, a_GCPtrMem) \
10839 do { \
10840 uint32_t u32Tmp; \
10841 IEM_MC_RETURN_ON_FAILURE(iemMemFetchDataU32(pVCpu, &u32Tmp, (a_iSeg), (a_GCPtrMem))); \
10842 (a_u64Dst) = (int32_t)u32Tmp; \
10843 } while (0)
10844#else /* IEM_WITH_SETJMP */
10845# define IEM_MC_FETCH_MEM_U8_SX_U16(a_u16Dst, a_iSeg, a_GCPtrMem) \
10846 ((a_u16Dst) = (int8_t)iemMemFetchDataU8Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10847# define IEM_MC_FETCH_MEM_U8_SX_U32(a_u32Dst, a_iSeg, a_GCPtrMem) \
10848 ((a_u32Dst) = (int8_t)iemMemFetchDataU8Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10849# define IEM_MC_FETCH_MEM_U8_SX_U64(a_u64Dst, a_iSeg, a_GCPtrMem) \
10850 ((a_u64Dst) = (int8_t)iemMemFetchDataU8Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10851# define IEM_MC_FETCH_MEM_U16_SX_U32(a_u32Dst, a_iSeg, a_GCPtrMem) \
10852 ((a_u32Dst) = (int16_t)iemMemFetchDataU16Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10853# define IEM_MC_FETCH_MEM_U16_SX_U64(a_u64Dst, a_iSeg, a_GCPtrMem) \
10854 ((a_u64Dst) = (int16_t)iemMemFetchDataU16Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10855# define IEM_MC_FETCH_MEM_U32_SX_U64(a_u64Dst, a_iSeg, a_GCPtrMem) \
10856 ((a_u64Dst) = (int32_t)iemMemFetchDataU32Jmp(pVCpu, (a_iSeg), (a_GCPtrMem)))
10857#endif /* IEM_WITH_SETJMP */
10858
10859#ifndef IEM_WITH_SETJMP
10860# define IEM_MC_STORE_MEM_U8(a_iSeg, a_GCPtrMem, a_u8Value) \
10861 IEM_MC_RETURN_ON_FAILURE(iemMemStoreDataU8(pVCpu, (a_iSeg), (a_GCPtrMem), (a_u8Value)))
10862# define IEM_MC_STORE_MEM_U16(a_iSeg, a_GCPtrMem, a_u16Value) \
10863 IEM_MC_RETURN_ON_FAILURE(iemMemStoreDataU16(pVCpu, (a_iSeg), (a_GCPtrMem), (a_u16Value)))
10864# define IEM_MC_STORE_MEM_U32(a_iSeg, a_GCPtrMem, a_u32Value) \
10865 IEM_MC_RETURN_ON_FAILURE(iemMemStoreDataU32(pVCpu, (a_iSeg), (a_GCPtrMem), (a_u32Value)))
10866# define IEM_MC_STORE_MEM_U64(a_iSeg, a_GCPtrMem, a_u64Value) \
10867 IEM_MC_RETURN_ON_FAILURE(iemMemStoreDataU64(pVCpu, (a_iSeg), (a_GCPtrMem), (a_u64Value)))
10868#else
10869# define IEM_MC_STORE_MEM_U8(a_iSeg, a_GCPtrMem, a_u8Value) \
10870 iemMemStoreDataU8Jmp(pVCpu, (a_iSeg), (a_GCPtrMem), (a_u8Value))
10871# define IEM_MC_STORE_MEM_U16(a_iSeg, a_GCPtrMem, a_u16Value) \
10872 iemMemStoreDataU16Jmp(pVCpu, (a_iSeg), (a_GCPtrMem), (a_u16Value))
10873# define IEM_MC_STORE_MEM_U32(a_iSeg, a_GCPtrMem, a_u32Value) \
10874 iemMemStoreDataU32Jmp(pVCpu, (a_iSeg), (a_GCPtrMem), (a_u32Value))
10875# define IEM_MC_STORE_MEM_U64(a_iSeg, a_GCPtrMem, a_u64Value) \
10876 iemMemStoreDataU64Jmp(pVCpu, (a_iSeg), (a_GCPtrMem), (a_u64Value))
10877#endif
10878
10879#ifndef IEM_WITH_SETJMP
10880# define IEM_MC_STORE_MEM_U8_CONST(a_iSeg, a_GCPtrMem, a_u8C) \
10881 IEM_MC_RETURN_ON_FAILURE(iemMemStoreDataU8(pVCpu, (a_iSeg), (a_GCPtrMem), (a_u8C)))
10882# define IEM_MC_STORE_MEM_U16_CONST(a_iSeg, a_GCPtrMem, a_u16C) \
10883 IEM_MC_RETURN_ON_FAILURE(iemMemStoreDataU16(pVCpu, (a_iSeg), (a_GCPtrMem), (a_u16C)))
10884# define IEM_MC_STORE_MEM_U32_CONST(a_iSeg, a_GCPtrMem, a_u32C) \
10885 IEM_MC_RETURN_ON_FAILURE(iemMemStoreDataU32(pVCpu, (a_iSeg), (a_GCPtrMem), (a_u32C)))
10886# define IEM_MC_STORE_MEM_U64_CONST(a_iSeg, a_GCPtrMem, a_u64C) \
10887 IEM_MC_RETURN_ON_FAILURE(iemMemStoreDataU64(pVCpu, (a_iSeg), (a_GCPtrMem), (a_u64C)))
10888#else
10889# define IEM_MC_STORE_MEM_U8_CONST(a_iSeg, a_GCPtrMem, a_u8C) \
10890 iemMemStoreDataU8Jmp(pVCpu, (a_iSeg), (a_GCPtrMem), (a_u8C))
10891# define IEM_MC_STORE_MEM_U16_CONST(a_iSeg, a_GCPtrMem, a_u16C) \
10892 iemMemStoreDataU16Jmp(pVCpu, (a_iSeg), (a_GCPtrMem), (a_u16C))
10893# define IEM_MC_STORE_MEM_U32_CONST(a_iSeg, a_GCPtrMem, a_u32C) \
10894 iemMemStoreDataU32Jmp(pVCpu, (a_iSeg), (a_GCPtrMem), (a_u32C))
10895# define IEM_MC_STORE_MEM_U64_CONST(a_iSeg, a_GCPtrMem, a_u64C) \
10896 iemMemStoreDataU64Jmp(pVCpu, (a_iSeg), (a_GCPtrMem), (a_u64C))
10897#endif
10898
10899#define IEM_MC_STORE_MEM_I8_CONST_BY_REF( a_pi8Dst, a_i8C) *(a_pi8Dst) = (a_i8C)
10900#define IEM_MC_STORE_MEM_I16_CONST_BY_REF(a_pi16Dst, a_i16C) *(a_pi16Dst) = (a_i16C)
10901#define IEM_MC_STORE_MEM_I32_CONST_BY_REF(a_pi32Dst, a_i32C) *(a_pi32Dst) = (a_i32C)
10902#define IEM_MC_STORE_MEM_I64_CONST_BY_REF(a_pi64Dst, a_i64C) *(a_pi64Dst) = (a_i64C)
10903#define IEM_MC_STORE_MEM_NEG_QNAN_R32_BY_REF(a_pr32Dst) (a_pr32Dst)->u32 = UINT32_C(0xffc00000)
10904#define IEM_MC_STORE_MEM_NEG_QNAN_R64_BY_REF(a_pr64Dst) (a_pr64Dst)->au64[0] = UINT64_C(0xfff8000000000000)
10905#define IEM_MC_STORE_MEM_NEG_QNAN_R80_BY_REF(a_pr80Dst) \
10906 do { \
10907 (a_pr80Dst)->au64[0] = UINT64_C(0xc000000000000000); \
10908 (a_pr80Dst)->au16[4] = UINT16_C(0xffff); \
10909 } while (0)
10910
10911#ifndef IEM_WITH_SETJMP
10912# define IEM_MC_STORE_MEM_U128(a_iSeg, a_GCPtrMem, a_u128Value) \
10913 IEM_MC_RETURN_ON_FAILURE(iemMemStoreDataU128(pVCpu, (a_iSeg), (a_GCPtrMem), (a_u128Value)))
10914# define IEM_MC_STORE_MEM_U128_ALIGN_SSE(a_iSeg, a_GCPtrMem, a_u128Value) \
10915 IEM_MC_RETURN_ON_FAILURE(iemMemStoreDataU128AlignedSse(pVCpu, (a_iSeg), (a_GCPtrMem), (a_u128Value)))
10916#else
10917# define IEM_MC_STORE_MEM_U128(a_iSeg, a_GCPtrMem, a_u128Value) \
10918 iemMemStoreDataU128Jmp(pVCpu, (a_iSeg), (a_GCPtrMem), (a_u128Value))
10919# define IEM_MC_STORE_MEM_U128_ALIGN_SSE(a_iSeg, a_GCPtrMem, a_u128Value) \
10920 iemMemStoreDataU128AlignedSseJmp(pVCpu, (a_iSeg), (a_GCPtrMem), (a_u128Value))
10921#endif
10922
10923
10924#define IEM_MC_PUSH_U16(a_u16Value) \
10925 IEM_MC_RETURN_ON_FAILURE(iemMemStackPushU16(pVCpu, (a_u16Value)))
10926#define IEM_MC_PUSH_U32(a_u32Value) \
10927 IEM_MC_RETURN_ON_FAILURE(iemMemStackPushU32(pVCpu, (a_u32Value)))
10928#define IEM_MC_PUSH_U32_SREG(a_u32Value) \
10929 IEM_MC_RETURN_ON_FAILURE(iemMemStackPushU32SReg(pVCpu, (a_u32Value)))
10930#define IEM_MC_PUSH_U64(a_u64Value) \
10931 IEM_MC_RETURN_ON_FAILURE(iemMemStackPushU64(pVCpu, (a_u64Value)))
10932
10933#define IEM_MC_POP_U16(a_pu16Value) \
10934 IEM_MC_RETURN_ON_FAILURE(iemMemStackPopU16(pVCpu, (a_pu16Value)))
10935#define IEM_MC_POP_U32(a_pu32Value) \
10936 IEM_MC_RETURN_ON_FAILURE(iemMemStackPopU32(pVCpu, (a_pu32Value)))
10937#define IEM_MC_POP_U64(a_pu64Value) \
10938 IEM_MC_RETURN_ON_FAILURE(iemMemStackPopU64(pVCpu, (a_pu64Value)))
10939
10940/** Maps guest memory for direct or bounce buffered access.
10941 * The purpose is to pass it to an operand implementation, thus the a_iArg.
10942 * @remarks May return.
10943 */
10944#define IEM_MC_MEM_MAP(a_pMem, a_fAccess, a_iSeg, a_GCPtrMem, a_iArg) \
10945 IEM_MC_RETURN_ON_FAILURE(iemMemMap(pVCpu, (void **)&(a_pMem), sizeof(*(a_pMem)), (a_iSeg), (a_GCPtrMem), (a_fAccess)))
10946
10947/** Maps guest memory for direct or bounce buffered access.
10948 * The purpose is to pass it to an operand implementation, thus the a_iArg.
10949 * @remarks May return.
10950 */
10951#define IEM_MC_MEM_MAP_EX(a_pvMem, a_fAccess, a_cbMem, a_iSeg, a_GCPtrMem, a_iArg) \
10952 IEM_MC_RETURN_ON_FAILURE(iemMemMap(pVCpu, (void **)&(a_pvMem), (a_cbMem), (a_iSeg), (a_GCPtrMem), (a_fAccess)))
10953
10954/** Commits the memory and unmaps the guest memory.
10955 * @remarks May return.
10956 */
10957#define IEM_MC_MEM_COMMIT_AND_UNMAP(a_pvMem, a_fAccess) \
10958 IEM_MC_RETURN_ON_FAILURE(iemMemCommitAndUnmap(pVCpu, (a_pvMem), (a_fAccess)))
10959
10960/** Commits the memory and unmaps the guest memory unless the FPU status word
10961 * indicates (@a a_u16FSW) and FPU control word indicates a pending exception
10962 * that would cause FLD not to store.
10963 *
10964 * The current understanding is that \#O, \#U, \#IA and \#IS will prevent a
10965 * store, while \#P will not.
10966 *
10967 * @remarks May in theory return - for now.
10968 */
10969#define IEM_MC_MEM_COMMIT_AND_UNMAP_FOR_FPU_STORE(a_pvMem, a_fAccess, a_u16FSW) \
10970 do { \
10971 if ( !(a_u16FSW & X86_FSW_ES) \
10972 || !( (a_u16FSW & (X86_FSW_UE | X86_FSW_OE | X86_FSW_IE)) \
10973 & ~(IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.FCW & X86_FCW_MASK_ALL) ) ) \
10974 IEM_MC_RETURN_ON_FAILURE(iemMemCommitAndUnmap(pVCpu, (a_pvMem), (a_fAccess))); \
10975 } while (0)
10976
10977/** Calculate efficient address from R/M. */
10978#ifndef IEM_WITH_SETJMP
10979# define IEM_MC_CALC_RM_EFF_ADDR(a_GCPtrEff, bRm, cbImm) \
10980 IEM_MC_RETURN_ON_FAILURE(iemOpHlpCalcRmEffAddr(pVCpu, (bRm), (cbImm), &(a_GCPtrEff)))
10981#else
10982# define IEM_MC_CALC_RM_EFF_ADDR(a_GCPtrEff, bRm, cbImm) \
10983 ((a_GCPtrEff) = iemOpHlpCalcRmEffAddrJmp(pVCpu, (bRm), (cbImm)))
10984#endif
10985
10986#define IEM_MC_CALL_VOID_AIMPL_0(a_pfn) (a_pfn)()
10987#define IEM_MC_CALL_VOID_AIMPL_1(a_pfn, a0) (a_pfn)((a0))
10988#define IEM_MC_CALL_VOID_AIMPL_2(a_pfn, a0, a1) (a_pfn)((a0), (a1))
10989#define IEM_MC_CALL_VOID_AIMPL_3(a_pfn, a0, a1, a2) (a_pfn)((a0), (a1), (a2))
10990#define IEM_MC_CALL_VOID_AIMPL_4(a_pfn, a0, a1, a2, a3) (a_pfn)((a0), (a1), (a2), (a3))
10991#define IEM_MC_CALL_AIMPL_3(a_rc, a_pfn, a0, a1, a2) (a_rc) = (a_pfn)((a0), (a1), (a2))
10992#define IEM_MC_CALL_AIMPL_4(a_rc, a_pfn, a0, a1, a2, a3) (a_rc) = (a_pfn)((a0), (a1), (a2), (a3))
10993
10994/**
10995 * Defers the rest of the instruction emulation to a C implementation routine
10996 * and returns, only taking the standard parameters.
10997 *
10998 * @param a_pfnCImpl The pointer to the C routine.
10999 * @sa IEM_DECL_IMPL_C_TYPE_0 and IEM_CIMPL_DEF_0.
11000 */
11001#define IEM_MC_CALL_CIMPL_0(a_pfnCImpl) return (a_pfnCImpl)(pVCpu, IEM_GET_INSTR_LEN(pVCpu))
11002
11003/**
11004 * Defers the rest of instruction emulation to a C implementation routine and
11005 * returns, taking one argument in addition to the standard ones.
11006 *
11007 * @param a_pfnCImpl The pointer to the C routine.
11008 * @param a0 The argument.
11009 */
11010#define IEM_MC_CALL_CIMPL_1(a_pfnCImpl, a0) return (a_pfnCImpl)(pVCpu, IEM_GET_INSTR_LEN(pVCpu), a0)
11011
11012/**
11013 * Defers the rest of the instruction emulation to a C implementation routine
11014 * and returns, taking two arguments in addition to the standard ones.
11015 *
11016 * @param a_pfnCImpl The pointer to the C routine.
11017 * @param a0 The first extra argument.
11018 * @param a1 The second extra argument.
11019 */
11020#define IEM_MC_CALL_CIMPL_2(a_pfnCImpl, a0, a1) return (a_pfnCImpl)(pVCpu, IEM_GET_INSTR_LEN(pVCpu), a0, a1)
11021
11022/**
11023 * Defers the rest of the instruction emulation to a C implementation routine
11024 * and returns, taking three arguments in addition to the standard ones.
11025 *
11026 * @param a_pfnCImpl The pointer to the C routine.
11027 * @param a0 The first extra argument.
11028 * @param a1 The second extra argument.
11029 * @param a2 The third extra argument.
11030 */
11031#define IEM_MC_CALL_CIMPL_3(a_pfnCImpl, a0, a1, a2) return (a_pfnCImpl)(pVCpu, IEM_GET_INSTR_LEN(pVCpu), a0, a1, a2)
11032
11033/**
11034 * Defers the rest of the instruction emulation to a C implementation routine
11035 * and returns, taking four arguments in addition to the standard ones.
11036 *
11037 * @param a_pfnCImpl The pointer to the C routine.
11038 * @param a0 The first extra argument.
11039 * @param a1 The second extra argument.
11040 * @param a2 The third extra argument.
11041 * @param a3 The fourth extra argument.
11042 */
11043#define IEM_MC_CALL_CIMPL_4(a_pfnCImpl, a0, a1, a2, a3) return (a_pfnCImpl)(pVCpu, IEM_GET_INSTR_LEN(pVCpu), a0, a1, a2, a3)
11044
11045/**
11046 * Defers the rest of the instruction emulation to a C implementation routine
11047 * and returns, taking two arguments in addition to the standard ones.
11048 *
11049 * @param a_pfnCImpl The pointer to the C routine.
11050 * @param a0 The first extra argument.
11051 * @param a1 The second extra argument.
11052 * @param a2 The third extra argument.
11053 * @param a3 The fourth extra argument.
11054 * @param a4 The fifth extra argument.
11055 */
11056#define IEM_MC_CALL_CIMPL_5(a_pfnCImpl, a0, a1, a2, a3, a4) return (a_pfnCImpl)(pVCpu, IEM_GET_INSTR_LEN(pVCpu), a0, a1, a2, a3, a4)
11057
11058/**
11059 * Defers the entire instruction emulation to a C implementation routine and
11060 * returns, only taking the standard parameters.
11061 *
11062 * This shall be used without any IEM_MC_BEGIN or IEM_END macro surrounding it.
11063 *
11064 * @param a_pfnCImpl The pointer to the C routine.
11065 * @sa IEM_DECL_IMPL_C_TYPE_0 and IEM_CIMPL_DEF_0.
11066 */
11067#define IEM_MC_DEFER_TO_CIMPL_0(a_pfnCImpl) (a_pfnCImpl)(pVCpu, IEM_GET_INSTR_LEN(pVCpu))
11068
11069/**
11070 * Defers the entire instruction emulation to a C implementation routine and
11071 * returns, taking one argument in addition to the standard ones.
11072 *
11073 * This shall be used without any IEM_MC_BEGIN or IEM_END macro surrounding it.
11074 *
11075 * @param a_pfnCImpl The pointer to the C routine.
11076 * @param a0 The argument.
11077 */
11078#define IEM_MC_DEFER_TO_CIMPL_1(a_pfnCImpl, a0) (a_pfnCImpl)(pVCpu, IEM_GET_INSTR_LEN(pVCpu), a0)
11079
11080/**
11081 * Defers the entire instruction emulation to a C implementation routine and
11082 * returns, taking two arguments in addition to the standard ones.
11083 *
11084 * This shall be used without any IEM_MC_BEGIN or IEM_END macro surrounding it.
11085 *
11086 * @param a_pfnCImpl The pointer to the C routine.
11087 * @param a0 The first extra argument.
11088 * @param a1 The second extra argument.
11089 */
11090#define IEM_MC_DEFER_TO_CIMPL_2(a_pfnCImpl, a0, a1) (a_pfnCImpl)(pVCpu, IEM_GET_INSTR_LEN(pVCpu), a0, a1)
11091
11092/**
11093 * Defers the entire instruction emulation to a C implementation routine and
11094 * returns, taking three arguments in addition to the standard ones.
11095 *
11096 * This shall be used without any IEM_MC_BEGIN or IEM_END macro surrounding it.
11097 *
11098 * @param a_pfnCImpl The pointer to the C routine.
11099 * @param a0 The first extra argument.
11100 * @param a1 The second extra argument.
11101 * @param a2 The third extra argument.
11102 */
11103#define IEM_MC_DEFER_TO_CIMPL_3(a_pfnCImpl, a0, a1, a2) (a_pfnCImpl)(pVCpu, IEM_GET_INSTR_LEN(pVCpu), a0, a1, a2)
11104
11105/**
11106 * Calls a FPU assembly implementation taking one visible argument.
11107 *
11108 * @param a_pfnAImpl Pointer to the assembly FPU routine.
11109 * @param a0 The first extra argument.
11110 */
11111#define IEM_MC_CALL_FPU_AIMPL_1(a_pfnAImpl, a0) \
11112 do { \
11113 a_pfnAImpl(&IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87, (a0)); \
11114 } while (0)
11115
11116/**
11117 * Calls a FPU assembly implementation taking two visible arguments.
11118 *
11119 * @param a_pfnAImpl Pointer to the assembly FPU routine.
11120 * @param a0 The first extra argument.
11121 * @param a1 The second extra argument.
11122 */
11123#define IEM_MC_CALL_FPU_AIMPL_2(a_pfnAImpl, a0, a1) \
11124 do { \
11125 a_pfnAImpl(&IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87, (a0), (a1)); \
11126 } while (0)
11127
11128/**
11129 * Calls a FPU assembly implementation taking three visible arguments.
11130 *
11131 * @param a_pfnAImpl Pointer to the assembly FPU routine.
11132 * @param a0 The first extra argument.
11133 * @param a1 The second extra argument.
11134 * @param a2 The third extra argument.
11135 */
11136#define IEM_MC_CALL_FPU_AIMPL_3(a_pfnAImpl, a0, a1, a2) \
11137 do { \
11138 a_pfnAImpl(&IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87, (a0), (a1), (a2)); \
11139 } while (0)
11140
11141#define IEM_MC_SET_FPU_RESULT(a_FpuData, a_FSW, a_pr80Value) \
11142 do { \
11143 (a_FpuData).FSW = (a_FSW); \
11144 (a_FpuData).r80Result = *(a_pr80Value); \
11145 } while (0)
11146
11147/** Pushes FPU result onto the stack. */
11148#define IEM_MC_PUSH_FPU_RESULT(a_FpuData) \
11149 iemFpuPushResult(pVCpu, &a_FpuData)
11150/** Pushes FPU result onto the stack and sets the FPUDP. */
11151#define IEM_MC_PUSH_FPU_RESULT_MEM_OP(a_FpuData, a_iEffSeg, a_GCPtrEff) \
11152 iemFpuPushResultWithMemOp(pVCpu, &a_FpuData, a_iEffSeg, a_GCPtrEff)
11153
11154/** Replaces ST0 with value one and pushes value 2 onto the FPU stack. */
11155#define IEM_MC_PUSH_FPU_RESULT_TWO(a_FpuDataTwo) \
11156 iemFpuPushResultTwo(pVCpu, &a_FpuDataTwo)
11157
11158/** Stores FPU result in a stack register. */
11159#define IEM_MC_STORE_FPU_RESULT(a_FpuData, a_iStReg) \
11160 iemFpuStoreResult(pVCpu, &a_FpuData, a_iStReg)
11161/** Stores FPU result in a stack register and pops the stack. */
11162#define IEM_MC_STORE_FPU_RESULT_THEN_POP(a_FpuData, a_iStReg) \
11163 iemFpuStoreResultThenPop(pVCpu, &a_FpuData, a_iStReg)
11164/** Stores FPU result in a stack register and sets the FPUDP. */
11165#define IEM_MC_STORE_FPU_RESULT_MEM_OP(a_FpuData, a_iStReg, a_iEffSeg, a_GCPtrEff) \
11166 iemFpuStoreResultWithMemOp(pVCpu, &a_FpuData, a_iStReg, a_iEffSeg, a_GCPtrEff)
11167/** Stores FPU result in a stack register, sets the FPUDP, and pops the
11168 * stack. */
11169#define IEM_MC_STORE_FPU_RESULT_WITH_MEM_OP_THEN_POP(a_FpuData, a_iStReg, a_iEffSeg, a_GCPtrEff) \
11170 iemFpuStoreResultWithMemOpThenPop(pVCpu, &a_FpuData, a_iStReg, a_iEffSeg, a_GCPtrEff)
11171
11172/** Only update the FOP, FPUIP, and FPUCS. (For FNOP.) */
11173#define IEM_MC_UPDATE_FPU_OPCODE_IP() \
11174 iemFpuUpdateOpcodeAndIp(pVCpu)
11175/** Free a stack register (for FFREE and FFREEP). */
11176#define IEM_MC_FPU_STACK_FREE(a_iStReg) \
11177 iemFpuStackFree(pVCpu, a_iStReg)
11178/** Increment the FPU stack pointer. */
11179#define IEM_MC_FPU_STACK_INC_TOP() \
11180 iemFpuStackIncTop(pVCpu)
11181/** Decrement the FPU stack pointer. */
11182#define IEM_MC_FPU_STACK_DEC_TOP() \
11183 iemFpuStackDecTop(pVCpu)
11184
11185/** Updates the FSW, FOP, FPUIP, and FPUCS. */
11186#define IEM_MC_UPDATE_FSW(a_u16FSW) \
11187 iemFpuUpdateFSW(pVCpu, a_u16FSW)
11188/** Updates the FSW with a constant value as well as FOP, FPUIP, and FPUCS. */
11189#define IEM_MC_UPDATE_FSW_CONST(a_u16FSW) \
11190 iemFpuUpdateFSW(pVCpu, a_u16FSW)
11191/** Updates the FSW, FOP, FPUIP, FPUCS, FPUDP, and FPUDS. */
11192#define IEM_MC_UPDATE_FSW_WITH_MEM_OP(a_u16FSW, a_iEffSeg, a_GCPtrEff) \
11193 iemFpuUpdateFSWWithMemOp(pVCpu, a_u16FSW, a_iEffSeg, a_GCPtrEff)
11194/** Updates the FSW, FOP, FPUIP, and FPUCS, and then pops the stack. */
11195#define IEM_MC_UPDATE_FSW_THEN_POP(a_u16FSW) \
11196 iemFpuUpdateFSWThenPop(pVCpu, a_u16FSW)
11197/** Updates the FSW, FOP, FPUIP, FPUCS, FPUDP and FPUDS, and then pops the
11198 * stack. */
11199#define IEM_MC_UPDATE_FSW_WITH_MEM_OP_THEN_POP(a_u16FSW, a_iEffSeg, a_GCPtrEff) \
11200 iemFpuUpdateFSWWithMemOpThenPop(pVCpu, a_u16FSW, a_iEffSeg, a_GCPtrEff)
11201/** Updates the FSW, FOP, FPUIP, and FPUCS, and then pops the stack twice. */
11202#define IEM_MC_UPDATE_FSW_THEN_POP_POP(a_u16FSW) \
11203 iemFpuUpdateFSWThenPop(pVCpu, a_u16FSW)
11204
11205/** Raises a FPU stack underflow exception. Sets FPUIP, FPUCS and FOP. */
11206#define IEM_MC_FPU_STACK_UNDERFLOW(a_iStDst) \
11207 iemFpuStackUnderflow(pVCpu, a_iStDst)
11208/** Raises a FPU stack underflow exception. Sets FPUIP, FPUCS and FOP. Pops
11209 * stack. */
11210#define IEM_MC_FPU_STACK_UNDERFLOW_THEN_POP(a_iStDst) \
11211 iemFpuStackUnderflowThenPop(pVCpu, a_iStDst)
11212/** Raises a FPU stack underflow exception. Sets FPUIP, FPUCS, FOP, FPUDP and
11213 * FPUDS. */
11214#define IEM_MC_FPU_STACK_UNDERFLOW_MEM_OP(a_iStDst, a_iEffSeg, a_GCPtrEff) \
11215 iemFpuStackUnderflowWithMemOp(pVCpu, a_iStDst, a_iEffSeg, a_GCPtrEff)
11216/** Raises a FPU stack underflow exception. Sets FPUIP, FPUCS, FOP, FPUDP and
11217 * FPUDS. Pops stack. */
11218#define IEM_MC_FPU_STACK_UNDERFLOW_MEM_OP_THEN_POP(a_iStDst, a_iEffSeg, a_GCPtrEff) \
11219 iemFpuStackUnderflowWithMemOpThenPop(pVCpu, a_iStDst, a_iEffSeg, a_GCPtrEff)
11220/** Raises a FPU stack underflow exception. Sets FPUIP, FPUCS and FOP. Pops
11221 * stack twice. */
11222#define IEM_MC_FPU_STACK_UNDERFLOW_THEN_POP_POP() \
11223 iemFpuStackUnderflowThenPopPop(pVCpu)
11224/** Raises a FPU stack underflow exception for an instruction pushing a result
11225 * value onto the stack. Sets FPUIP, FPUCS and FOP. */
11226#define IEM_MC_FPU_STACK_PUSH_UNDERFLOW() \
11227 iemFpuStackPushUnderflow(pVCpu)
11228/** Raises a FPU stack underflow exception for an instruction pushing a result
11229 * value onto the stack and replacing ST0. Sets FPUIP, FPUCS and FOP. */
11230#define IEM_MC_FPU_STACK_PUSH_UNDERFLOW_TWO() \
11231 iemFpuStackPushUnderflowTwo(pVCpu)
11232
11233/** Raises a FPU stack overflow exception as part of a push attempt. Sets
11234 * FPUIP, FPUCS and FOP. */
11235#define IEM_MC_FPU_STACK_PUSH_OVERFLOW() \
11236 iemFpuStackPushOverflow(pVCpu)
11237/** Raises a FPU stack overflow exception as part of a push attempt. Sets
11238 * FPUIP, FPUCS, FOP, FPUDP and FPUDS. */
11239#define IEM_MC_FPU_STACK_PUSH_OVERFLOW_MEM_OP(a_iEffSeg, a_GCPtrEff) \
11240 iemFpuStackPushOverflowWithMemOp(pVCpu, a_iEffSeg, a_GCPtrEff)
11241/** Prepares for using the FPU state.
11242 * Ensures that we can use the host FPU in the current context (RC+R0.
11243 * Ensures the guest FPU state in the CPUMCTX is up to date. */
11244#define IEM_MC_PREPARE_FPU_USAGE() iemFpuPrepareUsage(pVCpu)
11245/** Actualizes the guest FPU state so it can be accessed read-only fashion. */
11246#define IEM_MC_ACTUALIZE_FPU_STATE_FOR_READ() iemFpuActualizeStateForRead(pVCpu)
11247/** Actualizes the guest FPU state so it can be accessed and modified. */
11248#define IEM_MC_ACTUALIZE_FPU_STATE_FOR_CHANGE() iemFpuActualizeStateForChange(pVCpu)
11249
11250/** Prepares for using the SSE state.
11251 * Ensures that we can use the host SSE/FPU in the current context (RC+R0.
11252 * Ensures the guest SSE state in the CPUMCTX is up to date. */
11253#define IEM_MC_PREPARE_SSE_USAGE() iemFpuPrepareUsageSse(pVCpu)
11254/** Actualizes the guest XMM0..15 register state for read-only access. */
11255#define IEM_MC_ACTUALIZE_SSE_STATE_FOR_READ() iemFpuActualizeSseStateForRead(pVCpu)
11256/** Actualizes the guest XMM0..15 register state for read-write access. */
11257#define IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE() iemFpuActualizeSseStateForChange(pVCpu)
11258
11259/**
11260 * Calls a MMX assembly implementation taking two visible arguments.
11261 *
11262 * @param a_pfnAImpl Pointer to the assembly MMX routine.
11263 * @param a0 The first extra argument.
11264 * @param a1 The second extra argument.
11265 */
11266#define IEM_MC_CALL_MMX_AIMPL_2(a_pfnAImpl, a0, a1) \
11267 do { \
11268 IEM_MC_PREPARE_FPU_USAGE(); \
11269 a_pfnAImpl(&IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87, (a0), (a1)); \
11270 } while (0)
11271
11272/**
11273 * Calls a MMX assembly implementation taking three visible arguments.
11274 *
11275 * @param a_pfnAImpl Pointer to the assembly MMX routine.
11276 * @param a0 The first extra argument.
11277 * @param a1 The second extra argument.
11278 * @param a2 The third extra argument.
11279 */
11280#define IEM_MC_CALL_MMX_AIMPL_3(a_pfnAImpl, a0, a1, a2) \
11281 do { \
11282 IEM_MC_PREPARE_FPU_USAGE(); \
11283 a_pfnAImpl(&IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87, (a0), (a1), (a2)); \
11284 } while (0)
11285
11286
11287/**
11288 * Calls a SSE assembly implementation taking two visible arguments.
11289 *
11290 * @param a_pfnAImpl Pointer to the assembly MMX routine.
11291 * @param a0 The first extra argument.
11292 * @param a1 The second extra argument.
11293 */
11294#define IEM_MC_CALL_SSE_AIMPL_2(a_pfnAImpl, a0, a1) \
11295 do { \
11296 IEM_MC_PREPARE_SSE_USAGE(); \
11297 a_pfnAImpl(&IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87, (a0), (a1)); \
11298 } while (0)
11299
11300/**
11301 * Calls a SSE assembly implementation taking three visible arguments.
11302 *
11303 * @param a_pfnAImpl Pointer to the assembly MMX routine.
11304 * @param a0 The first extra argument.
11305 * @param a1 The second extra argument.
11306 * @param a2 The third extra argument.
11307 */
11308#define IEM_MC_CALL_SSE_AIMPL_3(a_pfnAImpl, a0, a1, a2) \
11309 do { \
11310 IEM_MC_PREPARE_SSE_USAGE(); \
11311 a_pfnAImpl(&IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87, (a0), (a1), (a2)); \
11312 } while (0)
11313
11314/** @note Not for IOPL or IF testing. */
11315#define IEM_MC_IF_EFL_BIT_SET(a_fBit) if (IEM_GET_CTX(pVCpu)->eflags.u & (a_fBit)) {
11316/** @note Not for IOPL or IF testing. */
11317#define IEM_MC_IF_EFL_BIT_NOT_SET(a_fBit) if (!(IEM_GET_CTX(pVCpu)->eflags.u & (a_fBit))) {
11318/** @note Not for IOPL or IF testing. */
11319#define IEM_MC_IF_EFL_ANY_BITS_SET(a_fBits) if (IEM_GET_CTX(pVCpu)->eflags.u & (a_fBits)) {
11320/** @note Not for IOPL or IF testing. */
11321#define IEM_MC_IF_EFL_NO_BITS_SET(a_fBits) if (!(IEM_GET_CTX(pVCpu)->eflags.u & (a_fBits))) {
11322/** @note Not for IOPL or IF testing. */
11323#define IEM_MC_IF_EFL_BITS_NE(a_fBit1, a_fBit2) \
11324 if ( !!(IEM_GET_CTX(pVCpu)->eflags.u & (a_fBit1)) \
11325 != !!(IEM_GET_CTX(pVCpu)->eflags.u & (a_fBit2)) ) {
11326/** @note Not for IOPL or IF testing. */
11327#define IEM_MC_IF_EFL_BITS_EQ(a_fBit1, a_fBit2) \
11328 if ( !!(IEM_GET_CTX(pVCpu)->eflags.u & (a_fBit1)) \
11329 == !!(IEM_GET_CTX(pVCpu)->eflags.u & (a_fBit2)) ) {
11330/** @note Not for IOPL or IF testing. */
11331#define IEM_MC_IF_EFL_BIT_SET_OR_BITS_NE(a_fBit, a_fBit1, a_fBit2) \
11332 if ( (IEM_GET_CTX(pVCpu)->eflags.u & (a_fBit)) \
11333 || !!(IEM_GET_CTX(pVCpu)->eflags.u & (a_fBit1)) \
11334 != !!(IEM_GET_CTX(pVCpu)->eflags.u & (a_fBit2)) ) {
11335/** @note Not for IOPL or IF testing. */
11336#define IEM_MC_IF_EFL_BIT_NOT_SET_AND_BITS_EQ(a_fBit, a_fBit1, a_fBit2) \
11337 if ( !(IEM_GET_CTX(pVCpu)->eflags.u & (a_fBit)) \
11338 && !!(IEM_GET_CTX(pVCpu)->eflags.u & (a_fBit1)) \
11339 == !!(IEM_GET_CTX(pVCpu)->eflags.u & (a_fBit2)) ) {
11340#define IEM_MC_IF_CX_IS_NZ() if (IEM_GET_CTX(pVCpu)->cx != 0) {
11341#define IEM_MC_IF_ECX_IS_NZ() if (IEM_GET_CTX(pVCpu)->ecx != 0) {
11342#define IEM_MC_IF_RCX_IS_NZ() if (IEM_GET_CTX(pVCpu)->rcx != 0) {
11343/** @note Not for IOPL or IF testing. */
11344#define IEM_MC_IF_CX_IS_NZ_AND_EFL_BIT_SET(a_fBit) \
11345 if ( IEM_GET_CTX(pVCpu)->cx != 0 \
11346 && (IEM_GET_CTX(pVCpu)->eflags.u & a_fBit)) {
11347/** @note Not for IOPL or IF testing. */
11348#define IEM_MC_IF_ECX_IS_NZ_AND_EFL_BIT_SET(a_fBit) \
11349 if ( IEM_GET_CTX(pVCpu)->ecx != 0 \
11350 && (IEM_GET_CTX(pVCpu)->eflags.u & a_fBit)) {
11351/** @note Not for IOPL or IF testing. */
11352#define IEM_MC_IF_RCX_IS_NZ_AND_EFL_BIT_SET(a_fBit) \
11353 if ( IEM_GET_CTX(pVCpu)->rcx != 0 \
11354 && (IEM_GET_CTX(pVCpu)->eflags.u & a_fBit)) {
11355/** @note Not for IOPL or IF testing. */
11356#define IEM_MC_IF_CX_IS_NZ_AND_EFL_BIT_NOT_SET(a_fBit) \
11357 if ( IEM_GET_CTX(pVCpu)->cx != 0 \
11358 && !(IEM_GET_CTX(pVCpu)->eflags.u & a_fBit)) {
11359/** @note Not for IOPL or IF testing. */
11360#define IEM_MC_IF_ECX_IS_NZ_AND_EFL_BIT_NOT_SET(a_fBit) \
11361 if ( IEM_GET_CTX(pVCpu)->ecx != 0 \
11362 && !(IEM_GET_CTX(pVCpu)->eflags.u & a_fBit)) {
11363/** @note Not for IOPL or IF testing. */
11364#define IEM_MC_IF_RCX_IS_NZ_AND_EFL_BIT_NOT_SET(a_fBit) \
11365 if ( IEM_GET_CTX(pVCpu)->rcx != 0 \
11366 && !(IEM_GET_CTX(pVCpu)->eflags.u & a_fBit)) {
11367#define IEM_MC_IF_LOCAL_IS_Z(a_Local) if ((a_Local) == 0) {
11368#define IEM_MC_IF_GREG_BIT_SET(a_iGReg, a_iBitNo) if (iemGRegFetchU64(pVCpu, (a_iGReg)) & RT_BIT_64(a_iBitNo)) {
11369
11370#define IEM_MC_IF_FPUREG_NOT_EMPTY(a_iSt) \
11371 if (iemFpuStRegNotEmpty(pVCpu, (a_iSt)) == VINF_SUCCESS) {
11372#define IEM_MC_IF_FPUREG_IS_EMPTY(a_iSt) \
11373 if (iemFpuStRegNotEmpty(pVCpu, (a_iSt)) != VINF_SUCCESS) {
11374#define IEM_MC_IF_FPUREG_NOT_EMPTY_REF_R80(a_pr80Dst, a_iSt) \
11375 if (iemFpuStRegNotEmptyRef(pVCpu, (a_iSt), &(a_pr80Dst)) == VINF_SUCCESS) {
11376#define IEM_MC_IF_TWO_FPUREGS_NOT_EMPTY_REF_R80(a_pr80Dst0, a_iSt0, a_pr80Dst1, a_iSt1) \
11377 if (iemFpu2StRegsNotEmptyRef(pVCpu, (a_iSt0), &(a_pr80Dst0), (a_iSt1), &(a_pr80Dst1)) == VINF_SUCCESS) {
11378#define IEM_MC_IF_TWO_FPUREGS_NOT_EMPTY_REF_R80_FIRST(a_pr80Dst0, a_iSt0, a_iSt1) \
11379 if (iemFpu2StRegsNotEmptyRefFirst(pVCpu, (a_iSt0), &(a_pr80Dst0), (a_iSt1)) == VINF_SUCCESS) {
11380#define IEM_MC_IF_FCW_IM() \
11381 if (IEM_GET_CTX(pVCpu)->CTX_SUFF(pXState)->x87.FCW & X86_FCW_IM) {
11382
11383#define IEM_MC_ELSE() } else {
11384#define IEM_MC_ENDIF() } do {} while (0)
11385
11386/** @} */
11387
11388
11389/** @name Opcode Debug Helpers.
11390 * @{
11391 */
11392#ifdef VBOX_WITH_STATISTICS
11393# define IEMOP_INC_STATS(a_Stats) do { pVCpu->iem.s.CTX_SUFF(pStats)->a_Stats += 1; } while (0)
11394#else
11395# define IEMOP_INC_STATS(a_Stats) do { } while (0)
11396#endif
11397
11398#ifdef DEBUG
11399# define IEMOP_MNEMONIC(a_Stats, a_szMnemonic) \
11400 do { \
11401 IEMOP_INC_STATS(a_Stats); \
11402 Log4(("decode - %04x:%RGv %s%s [#%u]\n", IEM_GET_CTX(pVCpu)->cs.Sel, IEM_GET_CTX(pVCpu)->rip, \
11403 pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK ? "lock " : "", a_szMnemonic, pVCpu->iem.s.cInstructions)); \
11404 } while (0)
11405#else
11406# define IEMOP_MNEMONIC(a_Stats, a_szMnemonic) IEMOP_INC_STATS(a_Stats)
11407#endif
11408
11409/** @} */
11410
11411
11412/** @name Opcode Helpers.
11413 * @{
11414 */
11415
11416#ifdef IN_RING3
11417# define IEMOP_HLP_MIN_CPU(a_uMinCpu, a_fOnlyIf) \
11418 do { \
11419 if (IEM_GET_TARGET_CPU(pVCpu) >= (a_uMinCpu) || !(a_fOnlyIf)) { } \
11420 else \
11421 { \
11422 (void)DBGFSTOP(pVCpu->CTX_SUFF(pVM)); \
11423 return IEMOP_RAISE_INVALID_OPCODE(); \
11424 } \
11425 } while (0)
11426#else
11427# define IEMOP_HLP_MIN_CPU(a_uMinCpu, a_fOnlyIf) \
11428 do { \
11429 if (IEM_GET_TARGET_CPU(pVCpu) >= (a_uMinCpu) || !(a_fOnlyIf)) { } \
11430 else return IEMOP_RAISE_INVALID_OPCODE(); \
11431 } while (0)
11432#endif
11433
11434/** The instruction requires a 186 or later. */
11435#if IEM_CFG_TARGET_CPU >= IEMTARGETCPU_186
11436# define IEMOP_HLP_MIN_186() do { } while (0)
11437#else
11438# define IEMOP_HLP_MIN_186() IEMOP_HLP_MIN_CPU(IEMTARGETCPU_186, true)
11439#endif
11440
11441/** The instruction requires a 286 or later. */
11442#if IEM_CFG_TARGET_CPU >= IEMTARGETCPU_286
11443# define IEMOP_HLP_MIN_286() do { } while (0)
11444#else
11445# define IEMOP_HLP_MIN_286() IEMOP_HLP_MIN_CPU(IEMTARGETCPU_286, true)
11446#endif
11447
11448/** The instruction requires a 386 or later. */
11449#if IEM_CFG_TARGET_CPU >= IEMTARGETCPU_386
11450# define IEMOP_HLP_MIN_386() do { } while (0)
11451#else
11452# define IEMOP_HLP_MIN_386() IEMOP_HLP_MIN_CPU(IEMTARGETCPU_386, true)
11453#endif
11454
11455/** The instruction requires a 386 or later if the given expression is true. */
11456#if IEM_CFG_TARGET_CPU >= IEMTARGETCPU_386
11457# define IEMOP_HLP_MIN_386_EX(a_fOnlyIf) do { } while (0)
11458#else
11459# define IEMOP_HLP_MIN_386_EX(a_fOnlyIf) IEMOP_HLP_MIN_CPU(IEMTARGETCPU_386, a_fOnlyIf)
11460#endif
11461
11462/** The instruction requires a 486 or later. */
11463#if IEM_CFG_TARGET_CPU >= IEMTARGETCPU_486
11464# define IEMOP_HLP_MIN_486() do { } while (0)
11465#else
11466# define IEMOP_HLP_MIN_486() IEMOP_HLP_MIN_CPU(IEMTARGETCPU_486, true)
11467#endif
11468
11469/** The instruction requires a Pentium (586) or later. */
11470#if IEM_CFG_TARGET_CPU >= IEMTARGETCPU_PENTIUM
11471# define IEMOP_HLP_MIN_586() do { } while (0)
11472#else
11473# define IEMOP_HLP_MIN_586() IEMOP_HLP_MIN_CPU(IEMTARGETCPU_PENTIUM, true)
11474#endif
11475
11476/** The instruction requires a PentiumPro (686) or later. */
11477#if IEM_CFG_TARGET_CPU >= IEMTARGETCPU_PPRO
11478# define IEMOP_HLP_MIN_686() do { } while (0)
11479#else
11480# define IEMOP_HLP_MIN_686() IEMOP_HLP_MIN_CPU(IEMTARGETCPU_PPRO, true)
11481#endif
11482
11483
11484/** The instruction raises an \#UD in real and V8086 mode. */
11485#define IEMOP_HLP_NO_REAL_OR_V86_MODE() \
11486 do \
11487 { \
11488 if (IEM_IS_REAL_OR_V86_MODE(pVCpu)) \
11489 return IEMOP_RAISE_INVALID_LOCK_PREFIX(); \
11490 } while (0)
11491
11492/** The instruction is not available in 64-bit mode, throw \#UD if we're in
11493 * 64-bit mode. */
11494#define IEMOP_HLP_NO_64BIT() \
11495 do \
11496 { \
11497 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT) \
11498 return IEMOP_RAISE_INVALID_OPCODE(); \
11499 } while (0)
11500
11501/** The instruction is only available in 64-bit mode, throw \#UD if we're not in
11502 * 64-bit mode. */
11503#define IEMOP_HLP_ONLY_64BIT() \
11504 do \
11505 { \
11506 if (pVCpu->iem.s.enmCpuMode != IEMMODE_64BIT) \
11507 return IEMOP_RAISE_INVALID_OPCODE(); \
11508 } while (0)
11509
11510/** The instruction defaults to 64-bit operand size if 64-bit mode. */
11511#define IEMOP_HLP_DEFAULT_64BIT_OP_SIZE() \
11512 do \
11513 { \
11514 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT) \
11515 iemRecalEffOpSize64Default(pVCpu); \
11516 } while (0)
11517
11518/** The instruction has 64-bit operand size if 64-bit mode. */
11519#define IEMOP_HLP_64BIT_OP_SIZE() \
11520 do \
11521 { \
11522 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT) \
11523 pVCpu->iem.s.enmEffOpSize = pVCpu->iem.s.enmDefOpSize = IEMMODE_64BIT; \
11524 } while (0)
11525
11526/** Only a REX prefix immediately preceeding the first opcode byte takes
11527 * effect. This macro helps ensuring this as well as logging bad guest code. */
11528#define IEMOP_HLP_CLEAR_REX_NOT_BEFORE_OPCODE(a_szPrf) \
11529 do \
11530 { \
11531 if (RT_UNLIKELY(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_REX)) \
11532 { \
11533 Log5((a_szPrf ": Overriding REX prefix at %RX16! fPrefixes=%#x\n", \
11534 IEM_GET_CTX(pVCpu)->rip, pVCpu->iem.s.fPrefixes)); \
11535 pVCpu->iem.s.fPrefixes &= ~IEM_OP_PRF_REX_MASK; \
11536 pVCpu->iem.s.uRexB = 0; \
11537 pVCpu->iem.s.uRexIndex = 0; \
11538 pVCpu->iem.s.uRexReg = 0; \
11539 iemRecalEffOpSize(pVCpu); \
11540 } \
11541 } while (0)
11542
11543/**
11544 * Done decoding.
11545 */
11546#define IEMOP_HLP_DONE_DECODING() \
11547 do \
11548 { \
11549 /*nothing for now, maybe later... */ \
11550 } while (0)
11551
11552/**
11553 * Done decoding, raise \#UD exception if lock prefix present.
11554 */
11555#define IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX() \
11556 do \
11557 { \
11558 if (RT_LIKELY(!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))) \
11559 { /* likely */ } \
11560 else \
11561 return IEMOP_RAISE_INVALID_LOCK_PREFIX(); \
11562 } while (0)
11563#define IEMOP_HLP_DECODED_NL_1(a_uDisOpNo, a_fIemOpFlags, a_uDisParam0, a_fDisOpType) \
11564 do \
11565 { \
11566 if (RT_LIKELY(!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))) \
11567 { /* likely */ } \
11568 else \
11569 { \
11570 NOREF(a_uDisOpNo); NOREF(a_fIemOpFlags); NOREF(a_uDisParam0); NOREF(a_fDisOpType); \
11571 return IEMOP_RAISE_INVALID_LOCK_PREFIX(); \
11572 } \
11573 } while (0)
11574#define IEMOP_HLP_DECODED_NL_2(a_uDisOpNo, a_fIemOpFlags, a_uDisParam0, a_uDisParam1, a_fDisOpType) \
11575 do \
11576 { \
11577 if (RT_LIKELY(!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))) \
11578 { /* likely */ } \
11579 else \
11580 { \
11581 NOREF(a_uDisOpNo); NOREF(a_fIemOpFlags); NOREF(a_uDisParam0); NOREF(a_uDisParam1); NOREF(a_fDisOpType); \
11582 return IEMOP_RAISE_INVALID_LOCK_PREFIX(); \
11583 } \
11584 } while (0)
11585
11586/**
11587 * Done decoding, raise \#UD exception if any lock, repz or repnz prefixes
11588 * are present.
11589 */
11590#define IEMOP_HLP_DONE_DECODING_NO_LOCK_REPZ_OR_REPNZ_PREFIXES() \
11591 do \
11592 { \
11593 if (RT_LIKELY(!(pVCpu->iem.s.fPrefixes & (IEM_OP_PRF_LOCK | IEM_OP_PRF_REPNZ | IEM_OP_PRF_REPZ)))) \
11594 { /* likely */ } \
11595 else \
11596 return IEMOP_RAISE_INVALID_OPCODE(); \
11597 } while (0)
11598
11599
11600/**
11601 * Calculates the effective address of a ModR/M memory operand.
11602 *
11603 * Meant to be used via IEM_MC_CALC_RM_EFF_ADDR.
11604 *
11605 * @return Strict VBox status code.
11606 * @param pVCpu The cross context virtual CPU structure of the calling thread.
11607 * @param bRm The ModRM byte.
11608 * @param cbImm The size of any immediate following the
11609 * effective address opcode bytes. Important for
11610 * RIP relative addressing.
11611 * @param pGCPtrEff Where to return the effective address.
11612 */
11613IEM_STATIC VBOXSTRICTRC iemOpHlpCalcRmEffAddr(PVMCPU pVCpu, uint8_t bRm, uint8_t cbImm, PRTGCPTR pGCPtrEff)
11614{
11615 Log5(("iemOpHlpCalcRmEffAddr: bRm=%#x\n", bRm));
11616 PCCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
11617# define SET_SS_DEF() \
11618 do \
11619 { \
11620 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_SEG_MASK)) \
11621 pVCpu->iem.s.iEffSeg = X86_SREG_SS; \
11622 } while (0)
11623
11624 if (pVCpu->iem.s.enmCpuMode != IEMMODE_64BIT)
11625 {
11626/** @todo Check the effective address size crap! */
11627 if (pVCpu->iem.s.enmEffAddrMode == IEMMODE_16BIT)
11628 {
11629 uint16_t u16EffAddr;
11630
11631 /* Handle the disp16 form with no registers first. */
11632 if ((bRm & (X86_MODRM_MOD_MASK | X86_MODRM_RM_MASK)) == 6)
11633 IEM_OPCODE_GET_NEXT_U16(&u16EffAddr);
11634 else
11635 {
11636 /* Get the displacment. */
11637 switch ((bRm >> X86_MODRM_MOD_SHIFT) & X86_MODRM_MOD_SMASK)
11638 {
11639 case 0: u16EffAddr = 0; break;
11640 case 1: IEM_OPCODE_GET_NEXT_S8_SX_U16(&u16EffAddr); break;
11641 case 2: IEM_OPCODE_GET_NEXT_U16(&u16EffAddr); break;
11642 default: AssertFailedReturn(VERR_IEM_IPE_1); /* (caller checked for these) */
11643 }
11644
11645 /* Add the base and index registers to the disp. */
11646 switch (bRm & X86_MODRM_RM_MASK)
11647 {
11648 case 0: u16EffAddr += pCtx->bx + pCtx->si; break;
11649 case 1: u16EffAddr += pCtx->bx + pCtx->di; break;
11650 case 2: u16EffAddr += pCtx->bp + pCtx->si; SET_SS_DEF(); break;
11651 case 3: u16EffAddr += pCtx->bp + pCtx->di; SET_SS_DEF(); break;
11652 case 4: u16EffAddr += pCtx->si; break;
11653 case 5: u16EffAddr += pCtx->di; break;
11654 case 6: u16EffAddr += pCtx->bp; SET_SS_DEF(); break;
11655 case 7: u16EffAddr += pCtx->bx; break;
11656 }
11657 }
11658
11659 *pGCPtrEff = u16EffAddr;
11660 }
11661 else
11662 {
11663 Assert(pVCpu->iem.s.enmEffAddrMode == IEMMODE_32BIT);
11664 uint32_t u32EffAddr;
11665
11666 /* Handle the disp32 form with no registers first. */
11667 if ((bRm & (X86_MODRM_MOD_MASK | X86_MODRM_RM_MASK)) == 5)
11668 IEM_OPCODE_GET_NEXT_U32(&u32EffAddr);
11669 else
11670 {
11671 /* Get the register (or SIB) value. */
11672 switch ((bRm & X86_MODRM_RM_MASK))
11673 {
11674 case 0: u32EffAddr = pCtx->eax; break;
11675 case 1: u32EffAddr = pCtx->ecx; break;
11676 case 2: u32EffAddr = pCtx->edx; break;
11677 case 3: u32EffAddr = pCtx->ebx; break;
11678 case 4: /* SIB */
11679 {
11680 uint8_t bSib; IEM_OPCODE_GET_NEXT_U8(&bSib);
11681
11682 /* Get the index and scale it. */
11683 switch ((bSib >> X86_SIB_INDEX_SHIFT) & X86_SIB_INDEX_SMASK)
11684 {
11685 case 0: u32EffAddr = pCtx->eax; break;
11686 case 1: u32EffAddr = pCtx->ecx; break;
11687 case 2: u32EffAddr = pCtx->edx; break;
11688 case 3: u32EffAddr = pCtx->ebx; break;
11689 case 4: u32EffAddr = 0; /*none */ break;
11690 case 5: u32EffAddr = pCtx->ebp; break;
11691 case 6: u32EffAddr = pCtx->esi; break;
11692 case 7: u32EffAddr = pCtx->edi; break;
11693 IEM_NOT_REACHED_DEFAULT_CASE_RET();
11694 }
11695 u32EffAddr <<= (bSib >> X86_SIB_SCALE_SHIFT) & X86_SIB_SCALE_SMASK;
11696
11697 /* add base */
11698 switch (bSib & X86_SIB_BASE_MASK)
11699 {
11700 case 0: u32EffAddr += pCtx->eax; break;
11701 case 1: u32EffAddr += pCtx->ecx; break;
11702 case 2: u32EffAddr += pCtx->edx; break;
11703 case 3: u32EffAddr += pCtx->ebx; break;
11704 case 4: u32EffAddr += pCtx->esp; SET_SS_DEF(); break;
11705 case 5:
11706 if ((bRm & X86_MODRM_MOD_MASK) != 0)
11707 {
11708 u32EffAddr += pCtx->ebp;
11709 SET_SS_DEF();
11710 }
11711 else
11712 {
11713 uint32_t u32Disp;
11714 IEM_OPCODE_GET_NEXT_U32(&u32Disp);
11715 u32EffAddr += u32Disp;
11716 }
11717 break;
11718 case 6: u32EffAddr += pCtx->esi; break;
11719 case 7: u32EffAddr += pCtx->edi; break;
11720 IEM_NOT_REACHED_DEFAULT_CASE_RET();
11721 }
11722 break;
11723 }
11724 case 5: u32EffAddr = pCtx->ebp; SET_SS_DEF(); break;
11725 case 6: u32EffAddr = pCtx->esi; break;
11726 case 7: u32EffAddr = pCtx->edi; break;
11727 IEM_NOT_REACHED_DEFAULT_CASE_RET();
11728 }
11729
11730 /* Get and add the displacement. */
11731 switch ((bRm >> X86_MODRM_MOD_SHIFT) & X86_MODRM_MOD_SMASK)
11732 {
11733 case 0:
11734 break;
11735 case 1:
11736 {
11737 int8_t i8Disp; IEM_OPCODE_GET_NEXT_S8(&i8Disp);
11738 u32EffAddr += i8Disp;
11739 break;
11740 }
11741 case 2:
11742 {
11743 uint32_t u32Disp; IEM_OPCODE_GET_NEXT_U32(&u32Disp);
11744 u32EffAddr += u32Disp;
11745 break;
11746 }
11747 default:
11748 AssertFailedReturn(VERR_IEM_IPE_2); /* (caller checked for these) */
11749 }
11750
11751 }
11752 if (pVCpu->iem.s.enmEffAddrMode == IEMMODE_32BIT)
11753 *pGCPtrEff = u32EffAddr;
11754 else
11755 {
11756 Assert(pVCpu->iem.s.enmEffAddrMode == IEMMODE_16BIT);
11757 *pGCPtrEff = u32EffAddr & UINT16_MAX;
11758 }
11759 }
11760 }
11761 else
11762 {
11763 uint64_t u64EffAddr;
11764
11765 /* Handle the rip+disp32 form with no registers first. */
11766 if ((bRm & (X86_MODRM_MOD_MASK | X86_MODRM_RM_MASK)) == 5)
11767 {
11768 IEM_OPCODE_GET_NEXT_S32_SX_U64(&u64EffAddr);
11769 u64EffAddr += pCtx->rip + IEM_GET_INSTR_LEN(pVCpu) + cbImm;
11770 }
11771 else
11772 {
11773 /* Get the register (or SIB) value. */
11774 switch ((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB)
11775 {
11776 case 0: u64EffAddr = pCtx->rax; break;
11777 case 1: u64EffAddr = pCtx->rcx; break;
11778 case 2: u64EffAddr = pCtx->rdx; break;
11779 case 3: u64EffAddr = pCtx->rbx; break;
11780 case 5: u64EffAddr = pCtx->rbp; SET_SS_DEF(); break;
11781 case 6: u64EffAddr = pCtx->rsi; break;
11782 case 7: u64EffAddr = pCtx->rdi; break;
11783 case 8: u64EffAddr = pCtx->r8; break;
11784 case 9: u64EffAddr = pCtx->r9; break;
11785 case 10: u64EffAddr = pCtx->r10; break;
11786 case 11: u64EffAddr = pCtx->r11; break;
11787 case 13: u64EffAddr = pCtx->r13; break;
11788 case 14: u64EffAddr = pCtx->r14; break;
11789 case 15: u64EffAddr = pCtx->r15; break;
11790 /* SIB */
11791 case 4:
11792 case 12:
11793 {
11794 uint8_t bSib; IEM_OPCODE_GET_NEXT_U8(&bSib);
11795
11796 /* Get the index and scale it. */
11797 switch (((bSib >> X86_SIB_INDEX_SHIFT) & X86_SIB_INDEX_SMASK) | pVCpu->iem.s.uRexIndex)
11798 {
11799 case 0: u64EffAddr = pCtx->rax; break;
11800 case 1: u64EffAddr = pCtx->rcx; break;
11801 case 2: u64EffAddr = pCtx->rdx; break;
11802 case 3: u64EffAddr = pCtx->rbx; break;
11803 case 4: u64EffAddr = 0; /*none */ break;
11804 case 5: u64EffAddr = pCtx->rbp; break;
11805 case 6: u64EffAddr = pCtx->rsi; break;
11806 case 7: u64EffAddr = pCtx->rdi; break;
11807 case 8: u64EffAddr = pCtx->r8; break;
11808 case 9: u64EffAddr = pCtx->r9; break;
11809 case 10: u64EffAddr = pCtx->r10; break;
11810 case 11: u64EffAddr = pCtx->r11; break;
11811 case 12: u64EffAddr = pCtx->r12; break;
11812 case 13: u64EffAddr = pCtx->r13; break;
11813 case 14: u64EffAddr = pCtx->r14; break;
11814 case 15: u64EffAddr = pCtx->r15; break;
11815 IEM_NOT_REACHED_DEFAULT_CASE_RET();
11816 }
11817 u64EffAddr <<= (bSib >> X86_SIB_SCALE_SHIFT) & X86_SIB_SCALE_SMASK;
11818
11819 /* add base */
11820 switch ((bSib & X86_SIB_BASE_MASK) | pVCpu->iem.s.uRexB)
11821 {
11822 case 0: u64EffAddr += pCtx->rax; break;
11823 case 1: u64EffAddr += pCtx->rcx; break;
11824 case 2: u64EffAddr += pCtx->rdx; break;
11825 case 3: u64EffAddr += pCtx->rbx; break;
11826 case 4: u64EffAddr += pCtx->rsp; SET_SS_DEF(); break;
11827 case 6: u64EffAddr += pCtx->rsi; break;
11828 case 7: u64EffAddr += pCtx->rdi; break;
11829 case 8: u64EffAddr += pCtx->r8; break;
11830 case 9: u64EffAddr += pCtx->r9; break;
11831 case 10: u64EffAddr += pCtx->r10; break;
11832 case 11: u64EffAddr += pCtx->r11; break;
11833 case 12: u64EffAddr += pCtx->r12; break;
11834 case 14: u64EffAddr += pCtx->r14; break;
11835 case 15: u64EffAddr += pCtx->r15; break;
11836 /* complicated encodings */
11837 case 5:
11838 case 13:
11839 if ((bRm & X86_MODRM_MOD_MASK) != 0)
11840 {
11841 if (!pVCpu->iem.s.uRexB)
11842 {
11843 u64EffAddr += pCtx->rbp;
11844 SET_SS_DEF();
11845 }
11846 else
11847 u64EffAddr += pCtx->r13;
11848 }
11849 else
11850 {
11851 uint32_t u32Disp;
11852 IEM_OPCODE_GET_NEXT_U32(&u32Disp);
11853 u64EffAddr += (int32_t)u32Disp;
11854 }
11855 break;
11856 IEM_NOT_REACHED_DEFAULT_CASE_RET();
11857 }
11858 break;
11859 }
11860 IEM_NOT_REACHED_DEFAULT_CASE_RET();
11861 }
11862
11863 /* Get and add the displacement. */
11864 switch ((bRm >> X86_MODRM_MOD_SHIFT) & X86_MODRM_MOD_SMASK)
11865 {
11866 case 0:
11867 break;
11868 case 1:
11869 {
11870 int8_t i8Disp;
11871 IEM_OPCODE_GET_NEXT_S8(&i8Disp);
11872 u64EffAddr += i8Disp;
11873 break;
11874 }
11875 case 2:
11876 {
11877 uint32_t u32Disp;
11878 IEM_OPCODE_GET_NEXT_U32(&u32Disp);
11879 u64EffAddr += (int32_t)u32Disp;
11880 break;
11881 }
11882 IEM_NOT_REACHED_DEFAULT_CASE_RET(); /* (caller checked for these) */
11883 }
11884
11885 }
11886
11887 if (pVCpu->iem.s.enmEffAddrMode == IEMMODE_64BIT)
11888 *pGCPtrEff = u64EffAddr;
11889 else
11890 {
11891 Assert(pVCpu->iem.s.enmEffAddrMode == IEMMODE_32BIT);
11892 *pGCPtrEff = u64EffAddr & UINT32_MAX;
11893 }
11894 }
11895
11896 Log5(("iemOpHlpCalcRmEffAddr: EffAddr=%#010RGv\n", *pGCPtrEff));
11897 return VINF_SUCCESS;
11898}
11899
11900
11901/**
11902 * Calculates the effective address of a ModR/M memory operand.
11903 *
11904 * Meant to be used via IEM_MC_CALC_RM_EFF_ADDR.
11905 *
11906 * @return Strict VBox status code.
11907 * @param pVCpu The cross context virtual CPU structure of the calling thread.
11908 * @param bRm The ModRM byte.
11909 * @param cbImm The size of any immediate following the
11910 * effective address opcode bytes. Important for
11911 * RIP relative addressing.
11912 * @param pGCPtrEff Where to return the effective address.
11913 * @param offRsp RSP displacement.
11914 */
11915IEM_STATIC VBOXSTRICTRC iemOpHlpCalcRmEffAddrEx(PVMCPU pVCpu, uint8_t bRm, uint8_t cbImm, PRTGCPTR pGCPtrEff, int8_t offRsp)
11916{
11917 Log5(("iemOpHlpCalcRmEffAddr: bRm=%#x\n", bRm));
11918 PCCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
11919# define SET_SS_DEF() \
11920 do \
11921 { \
11922 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_SEG_MASK)) \
11923 pVCpu->iem.s.iEffSeg = X86_SREG_SS; \
11924 } while (0)
11925
11926 if (pVCpu->iem.s.enmCpuMode != IEMMODE_64BIT)
11927 {
11928/** @todo Check the effective address size crap! */
11929 if (pVCpu->iem.s.enmEffAddrMode == IEMMODE_16BIT)
11930 {
11931 uint16_t u16EffAddr;
11932
11933 /* Handle the disp16 form with no registers first. */
11934 if ((bRm & (X86_MODRM_MOD_MASK | X86_MODRM_RM_MASK)) == 6)
11935 IEM_OPCODE_GET_NEXT_U16(&u16EffAddr);
11936 else
11937 {
11938 /* Get the displacment. */
11939 switch ((bRm >> X86_MODRM_MOD_SHIFT) & X86_MODRM_MOD_SMASK)
11940 {
11941 case 0: u16EffAddr = 0; break;
11942 case 1: IEM_OPCODE_GET_NEXT_S8_SX_U16(&u16EffAddr); break;
11943 case 2: IEM_OPCODE_GET_NEXT_U16(&u16EffAddr); break;
11944 default: AssertFailedReturn(VERR_IEM_IPE_1); /* (caller checked for these) */
11945 }
11946
11947 /* Add the base and index registers to the disp. */
11948 switch (bRm & X86_MODRM_RM_MASK)
11949 {
11950 case 0: u16EffAddr += pCtx->bx + pCtx->si; break;
11951 case 1: u16EffAddr += pCtx->bx + pCtx->di; break;
11952 case 2: u16EffAddr += pCtx->bp + pCtx->si; SET_SS_DEF(); break;
11953 case 3: u16EffAddr += pCtx->bp + pCtx->di; SET_SS_DEF(); break;
11954 case 4: u16EffAddr += pCtx->si; break;
11955 case 5: u16EffAddr += pCtx->di; break;
11956 case 6: u16EffAddr += pCtx->bp; SET_SS_DEF(); break;
11957 case 7: u16EffAddr += pCtx->bx; break;
11958 }
11959 }
11960
11961 *pGCPtrEff = u16EffAddr;
11962 }
11963 else
11964 {
11965 Assert(pVCpu->iem.s.enmEffAddrMode == IEMMODE_32BIT);
11966 uint32_t u32EffAddr;
11967
11968 /* Handle the disp32 form with no registers first. */
11969 if ((bRm & (X86_MODRM_MOD_MASK | X86_MODRM_RM_MASK)) == 5)
11970 IEM_OPCODE_GET_NEXT_U32(&u32EffAddr);
11971 else
11972 {
11973 /* Get the register (or SIB) value. */
11974 switch ((bRm & X86_MODRM_RM_MASK))
11975 {
11976 case 0: u32EffAddr = pCtx->eax; break;
11977 case 1: u32EffAddr = pCtx->ecx; break;
11978 case 2: u32EffAddr = pCtx->edx; break;
11979 case 3: u32EffAddr = pCtx->ebx; break;
11980 case 4: /* SIB */
11981 {
11982 uint8_t bSib; IEM_OPCODE_GET_NEXT_U8(&bSib);
11983
11984 /* Get the index and scale it. */
11985 switch ((bSib >> X86_SIB_INDEX_SHIFT) & X86_SIB_INDEX_SMASK)
11986 {
11987 case 0: u32EffAddr = pCtx->eax; break;
11988 case 1: u32EffAddr = pCtx->ecx; break;
11989 case 2: u32EffAddr = pCtx->edx; break;
11990 case 3: u32EffAddr = pCtx->ebx; break;
11991 case 4: u32EffAddr = 0; /*none */ break;
11992 case 5: u32EffAddr = pCtx->ebp; break;
11993 case 6: u32EffAddr = pCtx->esi; break;
11994 case 7: u32EffAddr = pCtx->edi; break;
11995 IEM_NOT_REACHED_DEFAULT_CASE_RET();
11996 }
11997 u32EffAddr <<= (bSib >> X86_SIB_SCALE_SHIFT) & X86_SIB_SCALE_SMASK;
11998
11999 /* add base */
12000 switch (bSib & X86_SIB_BASE_MASK)
12001 {
12002 case 0: u32EffAddr += pCtx->eax; break;
12003 case 1: u32EffAddr += pCtx->ecx; break;
12004 case 2: u32EffAddr += pCtx->edx; break;
12005 case 3: u32EffAddr += pCtx->ebx; break;
12006 case 4:
12007 u32EffAddr += pCtx->esp + offRsp;
12008 SET_SS_DEF();
12009 break;
12010 case 5:
12011 if ((bRm & X86_MODRM_MOD_MASK) != 0)
12012 {
12013 u32EffAddr += pCtx->ebp;
12014 SET_SS_DEF();
12015 }
12016 else
12017 {
12018 uint32_t u32Disp;
12019 IEM_OPCODE_GET_NEXT_U32(&u32Disp);
12020 u32EffAddr += u32Disp;
12021 }
12022 break;
12023 case 6: u32EffAddr += pCtx->esi; break;
12024 case 7: u32EffAddr += pCtx->edi; break;
12025 IEM_NOT_REACHED_DEFAULT_CASE_RET();
12026 }
12027 break;
12028 }
12029 case 5: u32EffAddr = pCtx->ebp; SET_SS_DEF(); break;
12030 case 6: u32EffAddr = pCtx->esi; break;
12031 case 7: u32EffAddr = pCtx->edi; break;
12032 IEM_NOT_REACHED_DEFAULT_CASE_RET();
12033 }
12034
12035 /* Get and add the displacement. */
12036 switch ((bRm >> X86_MODRM_MOD_SHIFT) & X86_MODRM_MOD_SMASK)
12037 {
12038 case 0:
12039 break;
12040 case 1:
12041 {
12042 int8_t i8Disp; IEM_OPCODE_GET_NEXT_S8(&i8Disp);
12043 u32EffAddr += i8Disp;
12044 break;
12045 }
12046 case 2:
12047 {
12048 uint32_t u32Disp; IEM_OPCODE_GET_NEXT_U32(&u32Disp);
12049 u32EffAddr += u32Disp;
12050 break;
12051 }
12052 default:
12053 AssertFailedReturn(VERR_IEM_IPE_2); /* (caller checked for these) */
12054 }
12055
12056 }
12057 if (pVCpu->iem.s.enmEffAddrMode == IEMMODE_32BIT)
12058 *pGCPtrEff = u32EffAddr;
12059 else
12060 {
12061 Assert(pVCpu->iem.s.enmEffAddrMode == IEMMODE_16BIT);
12062 *pGCPtrEff = u32EffAddr & UINT16_MAX;
12063 }
12064 }
12065 }
12066 else
12067 {
12068 uint64_t u64EffAddr;
12069
12070 /* Handle the rip+disp32 form with no registers first. */
12071 if ((bRm & (X86_MODRM_MOD_MASK | X86_MODRM_RM_MASK)) == 5)
12072 {
12073 IEM_OPCODE_GET_NEXT_S32_SX_U64(&u64EffAddr);
12074 u64EffAddr += pCtx->rip + IEM_GET_INSTR_LEN(pVCpu) + cbImm;
12075 }
12076 else
12077 {
12078 /* Get the register (or SIB) value. */
12079 switch ((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB)
12080 {
12081 case 0: u64EffAddr = pCtx->rax; break;
12082 case 1: u64EffAddr = pCtx->rcx; break;
12083 case 2: u64EffAddr = pCtx->rdx; break;
12084 case 3: u64EffAddr = pCtx->rbx; break;
12085 case 5: u64EffAddr = pCtx->rbp; SET_SS_DEF(); break;
12086 case 6: u64EffAddr = pCtx->rsi; break;
12087 case 7: u64EffAddr = pCtx->rdi; break;
12088 case 8: u64EffAddr = pCtx->r8; break;
12089 case 9: u64EffAddr = pCtx->r9; break;
12090 case 10: u64EffAddr = pCtx->r10; break;
12091 case 11: u64EffAddr = pCtx->r11; break;
12092 case 13: u64EffAddr = pCtx->r13; break;
12093 case 14: u64EffAddr = pCtx->r14; break;
12094 case 15: u64EffAddr = pCtx->r15; break;
12095 /* SIB */
12096 case 4:
12097 case 12:
12098 {
12099 uint8_t bSib; IEM_OPCODE_GET_NEXT_U8(&bSib);
12100
12101 /* Get the index and scale it. */
12102 switch (((bSib >> X86_SIB_INDEX_SHIFT) & X86_SIB_INDEX_SMASK) | pVCpu->iem.s.uRexIndex)
12103 {
12104 case 0: u64EffAddr = pCtx->rax; break;
12105 case 1: u64EffAddr = pCtx->rcx; break;
12106 case 2: u64EffAddr = pCtx->rdx; break;
12107 case 3: u64EffAddr = pCtx->rbx; break;
12108 case 4: u64EffAddr = 0; /*none */ break;
12109 case 5: u64EffAddr = pCtx->rbp; break;
12110 case 6: u64EffAddr = pCtx->rsi; break;
12111 case 7: u64EffAddr = pCtx->rdi; break;
12112 case 8: u64EffAddr = pCtx->r8; break;
12113 case 9: u64EffAddr = pCtx->r9; break;
12114 case 10: u64EffAddr = pCtx->r10; break;
12115 case 11: u64EffAddr = pCtx->r11; break;
12116 case 12: u64EffAddr = pCtx->r12; break;
12117 case 13: u64EffAddr = pCtx->r13; break;
12118 case 14: u64EffAddr = pCtx->r14; break;
12119 case 15: u64EffAddr = pCtx->r15; break;
12120 IEM_NOT_REACHED_DEFAULT_CASE_RET();
12121 }
12122 u64EffAddr <<= (bSib >> X86_SIB_SCALE_SHIFT) & X86_SIB_SCALE_SMASK;
12123
12124 /* add base */
12125 switch ((bSib & X86_SIB_BASE_MASK) | pVCpu->iem.s.uRexB)
12126 {
12127 case 0: u64EffAddr += pCtx->rax; break;
12128 case 1: u64EffAddr += pCtx->rcx; break;
12129 case 2: u64EffAddr += pCtx->rdx; break;
12130 case 3: u64EffAddr += pCtx->rbx; break;
12131 case 4: u64EffAddr += pCtx->rsp + offRsp; SET_SS_DEF(); break;
12132 case 6: u64EffAddr += pCtx->rsi; break;
12133 case 7: u64EffAddr += pCtx->rdi; break;
12134 case 8: u64EffAddr += pCtx->r8; break;
12135 case 9: u64EffAddr += pCtx->r9; break;
12136 case 10: u64EffAddr += pCtx->r10; break;
12137 case 11: u64EffAddr += pCtx->r11; break;
12138 case 12: u64EffAddr += pCtx->r12; break;
12139 case 14: u64EffAddr += pCtx->r14; break;
12140 case 15: u64EffAddr += pCtx->r15; break;
12141 /* complicated encodings */
12142 case 5:
12143 case 13:
12144 if ((bRm & X86_MODRM_MOD_MASK) != 0)
12145 {
12146 if (!pVCpu->iem.s.uRexB)
12147 {
12148 u64EffAddr += pCtx->rbp;
12149 SET_SS_DEF();
12150 }
12151 else
12152 u64EffAddr += pCtx->r13;
12153 }
12154 else
12155 {
12156 uint32_t u32Disp;
12157 IEM_OPCODE_GET_NEXT_U32(&u32Disp);
12158 u64EffAddr += (int32_t)u32Disp;
12159 }
12160 break;
12161 IEM_NOT_REACHED_DEFAULT_CASE_RET();
12162 }
12163 break;
12164 }
12165 IEM_NOT_REACHED_DEFAULT_CASE_RET();
12166 }
12167
12168 /* Get and add the displacement. */
12169 switch ((bRm >> X86_MODRM_MOD_SHIFT) & X86_MODRM_MOD_SMASK)
12170 {
12171 case 0:
12172 break;
12173 case 1:
12174 {
12175 int8_t i8Disp;
12176 IEM_OPCODE_GET_NEXT_S8(&i8Disp);
12177 u64EffAddr += i8Disp;
12178 break;
12179 }
12180 case 2:
12181 {
12182 uint32_t u32Disp;
12183 IEM_OPCODE_GET_NEXT_U32(&u32Disp);
12184 u64EffAddr += (int32_t)u32Disp;
12185 break;
12186 }
12187 IEM_NOT_REACHED_DEFAULT_CASE_RET(); /* (caller checked for these) */
12188 }
12189
12190 }
12191
12192 if (pVCpu->iem.s.enmEffAddrMode == IEMMODE_64BIT)
12193 *pGCPtrEff = u64EffAddr;
12194 else
12195 {
12196 Assert(pVCpu->iem.s.enmEffAddrMode == IEMMODE_32BIT);
12197 *pGCPtrEff = u64EffAddr & UINT32_MAX;
12198 }
12199 }
12200
12201 Log5(("iemOpHlpCalcRmEffAddr: EffAddr=%#010RGv\n", *pGCPtrEff));
12202 return VINF_SUCCESS;
12203}
12204
12205
12206#ifdef IEM_WITH_SETJMP
12207/**
12208 * Calculates the effective address of a ModR/M memory operand.
12209 *
12210 * Meant to be used via IEM_MC_CALC_RM_EFF_ADDR.
12211 *
12212 * May longjmp on internal error.
12213 *
12214 * @return The effective address.
12215 * @param pVCpu The cross context virtual CPU structure of the calling thread.
12216 * @param bRm The ModRM byte.
12217 * @param cbImm The size of any immediate following the
12218 * effective address opcode bytes. Important for
12219 * RIP relative addressing.
12220 */
12221IEM_STATIC RTGCPTR iemOpHlpCalcRmEffAddrJmp(PVMCPU pVCpu, uint8_t bRm, uint8_t cbImm)
12222{
12223 Log5(("iemOpHlpCalcRmEffAddrJmp: bRm=%#x\n", bRm));
12224 PCCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
12225# define SET_SS_DEF() \
12226 do \
12227 { \
12228 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_SEG_MASK)) \
12229 pVCpu->iem.s.iEffSeg = X86_SREG_SS; \
12230 } while (0)
12231
12232 if (pVCpu->iem.s.enmCpuMode != IEMMODE_64BIT)
12233 {
12234/** @todo Check the effective address size crap! */
12235 if (pVCpu->iem.s.enmEffAddrMode == IEMMODE_16BIT)
12236 {
12237 uint16_t u16EffAddr;
12238
12239 /* Handle the disp16 form with no registers first. */
12240 if ((bRm & (X86_MODRM_MOD_MASK | X86_MODRM_RM_MASK)) == 6)
12241 IEM_OPCODE_GET_NEXT_U16(&u16EffAddr);
12242 else
12243 {
12244 /* Get the displacment. */
12245 switch ((bRm >> X86_MODRM_MOD_SHIFT) & X86_MODRM_MOD_SMASK)
12246 {
12247 case 0: u16EffAddr = 0; break;
12248 case 1: IEM_OPCODE_GET_NEXT_S8_SX_U16(&u16EffAddr); break;
12249 case 2: IEM_OPCODE_GET_NEXT_U16(&u16EffAddr); break;
12250 default: AssertFailedStmt(longjmp(*pVCpu->iem.s.CTX_SUFF(pJmpBuf), VERR_IEM_IPE_1)); /* (caller checked for these) */
12251 }
12252
12253 /* Add the base and index registers to the disp. */
12254 switch (bRm & X86_MODRM_RM_MASK)
12255 {
12256 case 0: u16EffAddr += pCtx->bx + pCtx->si; break;
12257 case 1: u16EffAddr += pCtx->bx + pCtx->di; break;
12258 case 2: u16EffAddr += pCtx->bp + pCtx->si; SET_SS_DEF(); break;
12259 case 3: u16EffAddr += pCtx->bp + pCtx->di; SET_SS_DEF(); break;
12260 case 4: u16EffAddr += pCtx->si; break;
12261 case 5: u16EffAddr += pCtx->di; break;
12262 case 6: u16EffAddr += pCtx->bp; SET_SS_DEF(); break;
12263 case 7: u16EffAddr += pCtx->bx; break;
12264 }
12265 }
12266
12267 Log5(("iemOpHlpCalcRmEffAddrJmp: EffAddr=%#06RX16\n", u16EffAddr));
12268 return u16EffAddr;
12269 }
12270
12271 Assert(pVCpu->iem.s.enmEffAddrMode == IEMMODE_32BIT);
12272 uint32_t u32EffAddr;
12273
12274 /* Handle the disp32 form with no registers first. */
12275 if ((bRm & (X86_MODRM_MOD_MASK | X86_MODRM_RM_MASK)) == 5)
12276 IEM_OPCODE_GET_NEXT_U32(&u32EffAddr);
12277 else
12278 {
12279 /* Get the register (or SIB) value. */
12280 switch ((bRm & X86_MODRM_RM_MASK))
12281 {
12282 case 0: u32EffAddr = pCtx->eax; break;
12283 case 1: u32EffAddr = pCtx->ecx; break;
12284 case 2: u32EffAddr = pCtx->edx; break;
12285 case 3: u32EffAddr = pCtx->ebx; break;
12286 case 4: /* SIB */
12287 {
12288 uint8_t bSib; IEM_OPCODE_GET_NEXT_U8(&bSib);
12289
12290 /* Get the index and scale it. */
12291 switch ((bSib >> X86_SIB_INDEX_SHIFT) & X86_SIB_INDEX_SMASK)
12292 {
12293 case 0: u32EffAddr = pCtx->eax; break;
12294 case 1: u32EffAddr = pCtx->ecx; break;
12295 case 2: u32EffAddr = pCtx->edx; break;
12296 case 3: u32EffAddr = pCtx->ebx; break;
12297 case 4: u32EffAddr = 0; /*none */ break;
12298 case 5: u32EffAddr = pCtx->ebp; break;
12299 case 6: u32EffAddr = pCtx->esi; break;
12300 case 7: u32EffAddr = pCtx->edi; break;
12301 IEM_NOT_REACHED_DEFAULT_CASE_RET2(RTGCPTR_MAX);
12302 }
12303 u32EffAddr <<= (bSib >> X86_SIB_SCALE_SHIFT) & X86_SIB_SCALE_SMASK;
12304
12305 /* add base */
12306 switch (bSib & X86_SIB_BASE_MASK)
12307 {
12308 case 0: u32EffAddr += pCtx->eax; break;
12309 case 1: u32EffAddr += pCtx->ecx; break;
12310 case 2: u32EffAddr += pCtx->edx; break;
12311 case 3: u32EffAddr += pCtx->ebx; break;
12312 case 4: u32EffAddr += pCtx->esp; SET_SS_DEF(); break;
12313 case 5:
12314 if ((bRm & X86_MODRM_MOD_MASK) != 0)
12315 {
12316 u32EffAddr += pCtx->ebp;
12317 SET_SS_DEF();
12318 }
12319 else
12320 {
12321 uint32_t u32Disp;
12322 IEM_OPCODE_GET_NEXT_U32(&u32Disp);
12323 u32EffAddr += u32Disp;
12324 }
12325 break;
12326 case 6: u32EffAddr += pCtx->esi; break;
12327 case 7: u32EffAddr += pCtx->edi; break;
12328 IEM_NOT_REACHED_DEFAULT_CASE_RET2(RTGCPTR_MAX);
12329 }
12330 break;
12331 }
12332 case 5: u32EffAddr = pCtx->ebp; SET_SS_DEF(); break;
12333 case 6: u32EffAddr = pCtx->esi; break;
12334 case 7: u32EffAddr = pCtx->edi; break;
12335 IEM_NOT_REACHED_DEFAULT_CASE_RET2(RTGCPTR_MAX);
12336 }
12337
12338 /* Get and add the displacement. */
12339 switch ((bRm >> X86_MODRM_MOD_SHIFT) & X86_MODRM_MOD_SMASK)
12340 {
12341 case 0:
12342 break;
12343 case 1:
12344 {
12345 int8_t i8Disp; IEM_OPCODE_GET_NEXT_S8(&i8Disp);
12346 u32EffAddr += i8Disp;
12347 break;
12348 }
12349 case 2:
12350 {
12351 uint32_t u32Disp; IEM_OPCODE_GET_NEXT_U32(&u32Disp);
12352 u32EffAddr += u32Disp;
12353 break;
12354 }
12355 default:
12356 AssertFailedStmt(longjmp(*pVCpu->iem.s.CTX_SUFF(pJmpBuf), VERR_IEM_IPE_2)); /* (caller checked for these) */
12357 }
12358 }
12359
12360 if (pVCpu->iem.s.enmEffAddrMode == IEMMODE_32BIT)
12361 {
12362 Log5(("iemOpHlpCalcRmEffAddrJmp: EffAddr=%#010RX32\n", u32EffAddr));
12363 return u32EffAddr;
12364 }
12365 Assert(pVCpu->iem.s.enmEffAddrMode == IEMMODE_16BIT);
12366 Log5(("iemOpHlpCalcRmEffAddrJmp: EffAddr=%#06RX32\n", u32EffAddr & UINT16_MAX));
12367 return u32EffAddr & UINT16_MAX;
12368 }
12369
12370 uint64_t u64EffAddr;
12371
12372 /* Handle the rip+disp32 form with no registers first. */
12373 if ((bRm & (X86_MODRM_MOD_MASK | X86_MODRM_RM_MASK)) == 5)
12374 {
12375 IEM_OPCODE_GET_NEXT_S32_SX_U64(&u64EffAddr);
12376 u64EffAddr += pCtx->rip + IEM_GET_INSTR_LEN(pVCpu) + cbImm;
12377 }
12378 else
12379 {
12380 /* Get the register (or SIB) value. */
12381 switch ((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB)
12382 {
12383 case 0: u64EffAddr = pCtx->rax; break;
12384 case 1: u64EffAddr = pCtx->rcx; break;
12385 case 2: u64EffAddr = pCtx->rdx; break;
12386 case 3: u64EffAddr = pCtx->rbx; break;
12387 case 5: u64EffAddr = pCtx->rbp; SET_SS_DEF(); break;
12388 case 6: u64EffAddr = pCtx->rsi; break;
12389 case 7: u64EffAddr = pCtx->rdi; break;
12390 case 8: u64EffAddr = pCtx->r8; break;
12391 case 9: u64EffAddr = pCtx->r9; break;
12392 case 10: u64EffAddr = pCtx->r10; break;
12393 case 11: u64EffAddr = pCtx->r11; break;
12394 case 13: u64EffAddr = pCtx->r13; break;
12395 case 14: u64EffAddr = pCtx->r14; break;
12396 case 15: u64EffAddr = pCtx->r15; break;
12397 /* SIB */
12398 case 4:
12399 case 12:
12400 {
12401 uint8_t bSib; IEM_OPCODE_GET_NEXT_U8(&bSib);
12402
12403 /* Get the index and scale it. */
12404 switch (((bSib >> X86_SIB_INDEX_SHIFT) & X86_SIB_INDEX_SMASK) | pVCpu->iem.s.uRexIndex)
12405 {
12406 case 0: u64EffAddr = pCtx->rax; break;
12407 case 1: u64EffAddr = pCtx->rcx; break;
12408 case 2: u64EffAddr = pCtx->rdx; break;
12409 case 3: u64EffAddr = pCtx->rbx; break;
12410 case 4: u64EffAddr = 0; /*none */ break;
12411 case 5: u64EffAddr = pCtx->rbp; break;
12412 case 6: u64EffAddr = pCtx->rsi; break;
12413 case 7: u64EffAddr = pCtx->rdi; break;
12414 case 8: u64EffAddr = pCtx->r8; break;
12415 case 9: u64EffAddr = pCtx->r9; break;
12416 case 10: u64EffAddr = pCtx->r10; break;
12417 case 11: u64EffAddr = pCtx->r11; break;
12418 case 12: u64EffAddr = pCtx->r12; break;
12419 case 13: u64EffAddr = pCtx->r13; break;
12420 case 14: u64EffAddr = pCtx->r14; break;
12421 case 15: u64EffAddr = pCtx->r15; break;
12422 IEM_NOT_REACHED_DEFAULT_CASE_RET2(RTGCPTR_MAX);
12423 }
12424 u64EffAddr <<= (bSib >> X86_SIB_SCALE_SHIFT) & X86_SIB_SCALE_SMASK;
12425
12426 /* add base */
12427 switch ((bSib & X86_SIB_BASE_MASK) | pVCpu->iem.s.uRexB)
12428 {
12429 case 0: u64EffAddr += pCtx->rax; break;
12430 case 1: u64EffAddr += pCtx->rcx; break;
12431 case 2: u64EffAddr += pCtx->rdx; break;
12432 case 3: u64EffAddr += pCtx->rbx; break;
12433 case 4: u64EffAddr += pCtx->rsp; SET_SS_DEF(); break;
12434 case 6: u64EffAddr += pCtx->rsi; break;
12435 case 7: u64EffAddr += pCtx->rdi; break;
12436 case 8: u64EffAddr += pCtx->r8; break;
12437 case 9: u64EffAddr += pCtx->r9; break;
12438 case 10: u64EffAddr += pCtx->r10; break;
12439 case 11: u64EffAddr += pCtx->r11; break;
12440 case 12: u64EffAddr += pCtx->r12; break;
12441 case 14: u64EffAddr += pCtx->r14; break;
12442 case 15: u64EffAddr += pCtx->r15; break;
12443 /* complicated encodings */
12444 case 5:
12445 case 13:
12446 if ((bRm & X86_MODRM_MOD_MASK) != 0)
12447 {
12448 if (!pVCpu->iem.s.uRexB)
12449 {
12450 u64EffAddr += pCtx->rbp;
12451 SET_SS_DEF();
12452 }
12453 else
12454 u64EffAddr += pCtx->r13;
12455 }
12456 else
12457 {
12458 uint32_t u32Disp;
12459 IEM_OPCODE_GET_NEXT_U32(&u32Disp);
12460 u64EffAddr += (int32_t)u32Disp;
12461 }
12462 break;
12463 IEM_NOT_REACHED_DEFAULT_CASE_RET2(RTGCPTR_MAX);
12464 }
12465 break;
12466 }
12467 IEM_NOT_REACHED_DEFAULT_CASE_RET2(RTGCPTR_MAX);
12468 }
12469
12470 /* Get and add the displacement. */
12471 switch ((bRm >> X86_MODRM_MOD_SHIFT) & X86_MODRM_MOD_SMASK)
12472 {
12473 case 0:
12474 break;
12475 case 1:
12476 {
12477 int8_t i8Disp;
12478 IEM_OPCODE_GET_NEXT_S8(&i8Disp);
12479 u64EffAddr += i8Disp;
12480 break;
12481 }
12482 case 2:
12483 {
12484 uint32_t u32Disp;
12485 IEM_OPCODE_GET_NEXT_U32(&u32Disp);
12486 u64EffAddr += (int32_t)u32Disp;
12487 break;
12488 }
12489 IEM_NOT_REACHED_DEFAULT_CASE_RET2(RTGCPTR_MAX); /* (caller checked for these) */
12490 }
12491
12492 }
12493
12494 if (pVCpu->iem.s.enmEffAddrMode == IEMMODE_64BIT)
12495 {
12496 Log5(("iemOpHlpCalcRmEffAddrJmp: EffAddr=%#010RGv\n", u64EffAddr));
12497 return u64EffAddr;
12498 }
12499 Assert(pVCpu->iem.s.enmEffAddrMode == IEMMODE_32BIT);
12500 Log5(("iemOpHlpCalcRmEffAddrJmp: EffAddr=%#010RGv\n", u64EffAddr & UINT32_MAX));
12501 return u64EffAddr & UINT32_MAX;
12502}
12503#endif /* IEM_WITH_SETJMP */
12504
12505
12506/** @} */
12507
12508
12509
12510/*
12511 * Include the instructions
12512 */
12513#include "IEMAllInstructions.cpp.h"
12514
12515
12516
12517
12518#if defined(IEM_VERIFICATION_MODE_FULL) && defined(IN_RING3)
12519
12520/**
12521 * Sets up execution verification mode.
12522 */
12523IEM_STATIC void iemExecVerificationModeSetup(PVMCPU pVCpu)
12524{
12525 PVMCPU pVCpu = pVCpu;
12526 PCPUMCTX pOrgCtx = IEM_GET_CTX(pVCpu);
12527
12528 /*
12529 * Always note down the address of the current instruction.
12530 */
12531 pVCpu->iem.s.uOldCs = pOrgCtx->cs.Sel;
12532 pVCpu->iem.s.uOldRip = pOrgCtx->rip;
12533
12534 /*
12535 * Enable verification and/or logging.
12536 */
12537 bool fNewNoRem = !LogIs6Enabled(); /* logging triggers the no-rem/rem verification stuff */;
12538 if ( fNewNoRem
12539 && ( 0
12540#if 0 /* auto enable on first paged protected mode interrupt */
12541 || ( pOrgCtx->eflags.Bits.u1IF
12542 && (pOrgCtx->cr0 & (X86_CR0_PE | X86_CR0_PG)) == (X86_CR0_PE | X86_CR0_PG)
12543 && TRPMHasTrap(pVCpu)
12544 && EMGetInhibitInterruptsPC(pVCpu) != pOrgCtx->rip) )
12545#endif
12546#if 0
12547 || ( pOrgCtx->cs == 0x10
12548 && ( pOrgCtx->rip == 0x90119e3e
12549 || pOrgCtx->rip == 0x901d9810)
12550#endif
12551#if 0 /* Auto enable DSL - FPU stuff. */
12552 || ( pOrgCtx->cs == 0x10
12553 && (// pOrgCtx->rip == 0xc02ec07f
12554 //|| pOrgCtx->rip == 0xc02ec082
12555 //|| pOrgCtx->rip == 0xc02ec0c9
12556 0
12557 || pOrgCtx->rip == 0x0c010e7c4 /* fxsave */ ) )
12558#endif
12559#if 0 /* Auto enable DSL - fstp st0 stuff. */
12560 || (pOrgCtx->cs.Sel == 0x23 pOrgCtx->rip == 0x804aff7)
12561#endif
12562#if 0
12563 || pOrgCtx->rip == 0x9022bb3a
12564#endif
12565#if 0
12566 || (pOrgCtx->cs.Sel == 0x58 && pOrgCtx->rip == 0x3be) /* NT4SP1 sidt/sgdt in early loader code */
12567#endif
12568#if 0
12569 || (pOrgCtx->cs.Sel == 8 && pOrgCtx->rip == 0x8013ec28) /* NT4SP1 first str (early boot) */
12570 || (pOrgCtx->cs.Sel == 8 && pOrgCtx->rip == 0x80119e3f) /* NT4SP1 second str (early boot) */
12571#endif
12572#if 0 /* NT4SP1 - later on the blue screen, things goes wrong... */
12573 || (pOrgCtx->cs.Sel == 8 && pOrgCtx->rip == 0x8010a5df)
12574 || (pOrgCtx->cs.Sel == 8 && pOrgCtx->rip == 0x8013a7c4)
12575 || (pOrgCtx->cs.Sel == 8 && pOrgCtx->rip == 0x8013a7d2)
12576#endif
12577#if 0 /* NT4SP1 - xadd early boot. */
12578 || (pOrgCtx->cs.Sel == 8 && pOrgCtx->rip == 0x8019cf0f)
12579#endif
12580#if 0 /* NT4SP1 - wrmsr (intel MSR). */
12581 || (pOrgCtx->cs.Sel == 8 && pOrgCtx->rip == 0x8011a6d4)
12582#endif
12583#if 0 /* NT4SP1 - cmpxchg (AMD). */
12584 || (pOrgCtx->cs.Sel == 8 && pOrgCtx->rip == 0x801684c1)
12585#endif
12586#if 0 /* NT4SP1 - fnstsw + 2 (AMD). */
12587 || (pOrgCtx->cs.Sel == 8 && pOrgCtx->rip == 0x801c6b88+2)
12588#endif
12589#if 0 /* NT4SP1 - iret to v8086 -- too generic a place? (N/A with GAs installed) */
12590 || (pOrgCtx->cs.Sel == 8 && pOrgCtx->rip == 0x8013bd5d)
12591
12592#endif
12593#if 0 /* NT4SP1 - iret to v8086 (executing edlin) */
12594 || (pOrgCtx->cs.Sel == 8 && pOrgCtx->rip == 0x8013b609)
12595
12596#endif
12597#if 0 /* NT4SP1 - frstor [ecx] */
12598 || (pOrgCtx->cs.Sel == 8 && pOrgCtx->rip == 0x8013d11f)
12599#endif
12600#if 0 /* xxxxxx - All long mode code. */
12601 || (pOrgCtx->msrEFER & MSR_K6_EFER_LMA)
12602#endif
12603#if 0 /* rep movsq linux 3.7 64-bit boot. */
12604 || (pOrgCtx->rip == 0x0000000000100241)
12605#endif
12606#if 0 /* linux 3.7 64-bit boot - '000000000215e240'. */
12607 || (pOrgCtx->rip == 0x000000000215e240)
12608#endif
12609#if 0 /* DOS's size-overridden iret to v8086. */
12610 || (pOrgCtx->rip == 0x427 && pOrgCtx->cs.Sel == 0xb8)
12611#endif
12612 )
12613 )
12614 {
12615 RTLogGroupSettings(NULL, "iem.eo.l6.l2");
12616 RTLogFlags(NULL, "enabled");
12617 fNewNoRem = false;
12618 }
12619 if (fNewNoRem != pVCpu->iem.s.fNoRem)
12620 {
12621 pVCpu->iem.s.fNoRem = fNewNoRem;
12622 if (!fNewNoRem)
12623 {
12624 LogAlways(("Enabling verification mode!\n"));
12625 CPUMSetChangedFlags(pVCpu, CPUM_CHANGED_ALL);
12626 }
12627 else
12628 LogAlways(("Disabling verification mode!\n"));
12629 }
12630
12631 /*
12632 * Switch state.
12633 */
12634 if (IEM_VERIFICATION_ENABLED(pVCpu))
12635 {
12636 static CPUMCTX s_DebugCtx; /* Ugly! */
12637
12638 s_DebugCtx = *pOrgCtx;
12639 IEM_GET_CTX(pVCpu) = &s_DebugCtx;
12640 }
12641
12642 /*
12643 * See if there is an interrupt pending in TRPM and inject it if we can.
12644 */
12645 pVCpu->iem.s.uInjectCpl = UINT8_MAX;
12646 if ( pOrgCtx->eflags.Bits.u1IF
12647 && TRPMHasTrap(pVCpu)
12648 && EMGetInhibitInterruptsPC(pVCpu) != pOrgCtx->rip)
12649 {
12650 uint8_t u8TrapNo;
12651 TRPMEVENT enmType;
12652 RTGCUINT uErrCode;
12653 RTGCPTR uCr2;
12654 int rc2 = TRPMQueryTrapAll(pVCpu, &u8TrapNo, &enmType, &uErrCode, &uCr2, NULL /* pu8InstLen */); AssertRC(rc2);
12655 IEMInjectTrap(pVCpu, u8TrapNo, enmType, (uint16_t)uErrCode, uCr2, 0 /* cbInstr */);
12656 if (!IEM_VERIFICATION_ENABLED(pVCpu))
12657 TRPMResetTrap(pVCpu);
12658 pVCpu->iem.s.uInjectCpl = pVCpu->iem.s.uCpl;
12659 }
12660
12661 /*
12662 * Reset the counters.
12663 */
12664 pVCpu->iem.s.cIOReads = 0;
12665 pVCpu->iem.s.cIOWrites = 0;
12666 pVCpu->iem.s.fIgnoreRaxRdx = false;
12667 pVCpu->iem.s.fOverlappingMovs = false;
12668 pVCpu->iem.s.fProblematicMemory = false;
12669 pVCpu->iem.s.fUndefinedEFlags = 0;
12670
12671 if (IEM_VERIFICATION_ENABLED(pVCpu))
12672 {
12673 /*
12674 * Free all verification records.
12675 */
12676 PIEMVERIFYEVTREC pEvtRec = pVCpu->iem.s.pIemEvtRecHead;
12677 pVCpu->iem.s.pIemEvtRecHead = NULL;
12678 pVCpu->iem.s.ppIemEvtRecNext = &pVCpu->iem.s.pIemEvtRecHead;
12679 do
12680 {
12681 while (pEvtRec)
12682 {
12683 PIEMVERIFYEVTREC pNext = pEvtRec->pNext;
12684 pEvtRec->pNext = pVCpu->iem.s.pFreeEvtRec;
12685 pVCpu->iem.s.pFreeEvtRec = pEvtRec;
12686 pEvtRec = pNext;
12687 }
12688 pEvtRec = pVCpu->iem.s.pOtherEvtRecHead;
12689 pVCpu->iem.s.pOtherEvtRecHead = NULL;
12690 pVCpu->iem.s.ppOtherEvtRecNext = &pVCpu->iem.s.pOtherEvtRecHead;
12691 } while (pEvtRec);
12692 }
12693}
12694
12695
12696/**
12697 * Allocate an event record.
12698 * @returns Pointer to a record.
12699 */
12700IEM_STATIC PIEMVERIFYEVTREC iemVerifyAllocRecord(PVMCPU pVCpu)
12701{
12702 if (!IEM_VERIFICATION_ENABLED(pVCpu))
12703 return NULL;
12704
12705 PIEMVERIFYEVTREC pEvtRec = pVCpu->iem.s.pFreeEvtRec;
12706 if (pEvtRec)
12707 pVCpu->iem.s.pFreeEvtRec = pEvtRec->pNext;
12708 else
12709 {
12710 if (!pVCpu->iem.s.ppIemEvtRecNext)
12711 return NULL; /* Too early (fake PCIBIOS), ignore notification. */
12712
12713 pEvtRec = (PIEMVERIFYEVTREC)MMR3HeapAlloc(pVCpu->CTX_SUFF(pVM), MM_TAG_EM /* lazy bird*/, sizeof(*pEvtRec));
12714 if (!pEvtRec)
12715 return NULL;
12716 }
12717 pEvtRec->enmEvent = IEMVERIFYEVENT_INVALID;
12718 pEvtRec->pNext = NULL;
12719 return pEvtRec;
12720}
12721
12722
12723/**
12724 * IOMMMIORead notification.
12725 */
12726VMM_INT_DECL(void) IEMNotifyMMIORead(PVM pVM, RTGCPHYS GCPhys, size_t cbValue)
12727{
12728 PVMCPU pVCpu = VMMGetCpu(pVM);
12729 if (!pVCpu)
12730 return;
12731 PIEMVERIFYEVTREC pEvtRec = iemVerifyAllocRecord(pVCpu);
12732 if (!pEvtRec)
12733 return;
12734 pEvtRec->enmEvent = IEMVERIFYEVENT_RAM_READ;
12735 pEvtRec->u.RamRead.GCPhys = GCPhys;
12736 pEvtRec->u.RamRead.cb = (uint32_t)cbValue;
12737 pEvtRec->pNext = *pVCpu->iem.s.ppOtherEvtRecNext;
12738 *pVCpu->iem.s.ppOtherEvtRecNext = pEvtRec;
12739}
12740
12741
12742/**
12743 * IOMMMIOWrite notification.
12744 */
12745VMM_INT_DECL(void) IEMNotifyMMIOWrite(PVM pVM, RTGCPHYS GCPhys, uint32_t u32Value, size_t cbValue)
12746{
12747 PVMCPU pVCpu = VMMGetCpu(pVM);
12748 if (!pVCpu)
12749 return;
12750 PIEMVERIFYEVTREC pEvtRec = iemVerifyAllocRecord(pVCpu);
12751 if (!pEvtRec)
12752 return;
12753 pEvtRec->enmEvent = IEMVERIFYEVENT_RAM_WRITE;
12754 pEvtRec->u.RamWrite.GCPhys = GCPhys;
12755 pEvtRec->u.RamWrite.cb = (uint32_t)cbValue;
12756 pEvtRec->u.RamWrite.ab[0] = RT_BYTE1(u32Value);
12757 pEvtRec->u.RamWrite.ab[1] = RT_BYTE2(u32Value);
12758 pEvtRec->u.RamWrite.ab[2] = RT_BYTE3(u32Value);
12759 pEvtRec->u.RamWrite.ab[3] = RT_BYTE4(u32Value);
12760 pEvtRec->pNext = *pVCpu->iem.s.ppOtherEvtRecNext;
12761 *pVCpu->iem.s.ppOtherEvtRecNext = pEvtRec;
12762}
12763
12764
12765/**
12766 * IOMIOPortRead notification.
12767 */
12768VMM_INT_DECL(void) IEMNotifyIOPortRead(PVM pVM, RTIOPORT Port, size_t cbValue)
12769{
12770 PVMCPU pVCpu = VMMGetCpu(pVM);
12771 if (!pVCpu)
12772 return;
12773 PIEMVERIFYEVTREC pEvtRec = iemVerifyAllocRecord(pVCpu);
12774 if (!pEvtRec)
12775 return;
12776 pEvtRec->enmEvent = IEMVERIFYEVENT_IOPORT_READ;
12777 pEvtRec->u.IOPortRead.Port = Port;
12778 pEvtRec->u.IOPortRead.cbValue = (uint8_t)cbValue;
12779 pEvtRec->pNext = *pVCpu->iem.s.ppOtherEvtRecNext;
12780 *pVCpu->iem.s.ppOtherEvtRecNext = pEvtRec;
12781}
12782
12783/**
12784 * IOMIOPortWrite notification.
12785 */
12786VMM_INT_DECL(void) IEMNotifyIOPortWrite(PVM pVM, RTIOPORT Port, uint32_t u32Value, size_t cbValue)
12787{
12788 PVMCPU pVCpu = VMMGetCpu(pVM);
12789 if (!pVCpu)
12790 return;
12791 PIEMVERIFYEVTREC pEvtRec = iemVerifyAllocRecord(pVCpu);
12792 if (!pEvtRec)
12793 return;
12794 pEvtRec->enmEvent = IEMVERIFYEVENT_IOPORT_WRITE;
12795 pEvtRec->u.IOPortWrite.Port = Port;
12796 pEvtRec->u.IOPortWrite.cbValue = (uint8_t)cbValue;
12797 pEvtRec->u.IOPortWrite.u32Value = u32Value;
12798 pEvtRec->pNext = *pVCpu->iem.s.ppOtherEvtRecNext;
12799 *pVCpu->iem.s.ppOtherEvtRecNext = pEvtRec;
12800}
12801
12802
12803VMM_INT_DECL(void) IEMNotifyIOPortReadString(PVM pVM, RTIOPORT Port, void *pvDst, RTGCUINTREG cTransfers, size_t cbValue)
12804{
12805 PVMCPU pVCpu = VMMGetCpu(pVM);
12806 if (!pVCpu)
12807 return;
12808 PIEMVERIFYEVTREC pEvtRec = iemVerifyAllocRecord(pVCpu);
12809 if (!pEvtRec)
12810 return;
12811 pEvtRec->enmEvent = IEMVERIFYEVENT_IOPORT_STR_READ;
12812 pEvtRec->u.IOPortStrRead.Port = Port;
12813 pEvtRec->u.IOPortStrRead.cbValue = (uint8_t)cbValue;
12814 pEvtRec->u.IOPortStrRead.cTransfers = cTransfers;
12815 pEvtRec->pNext = *pVCpu->iem.s.ppOtherEvtRecNext;
12816 *pVCpu->iem.s.ppOtherEvtRecNext = pEvtRec;
12817}
12818
12819
12820VMM_INT_DECL(void) IEMNotifyIOPortWriteString(PVM pVM, RTIOPORT Port, void const *pvSrc, RTGCUINTREG cTransfers, size_t cbValue)
12821{
12822 PVMCPU pVCpu = VMMGetCpu(pVM);
12823 if (!pVCpu)
12824 return;
12825 PIEMVERIFYEVTREC pEvtRec = iemVerifyAllocRecord(pVCpu);
12826 if (!pEvtRec)
12827 return;
12828 pEvtRec->enmEvent = IEMVERIFYEVENT_IOPORT_STR_WRITE;
12829 pEvtRec->u.IOPortStrWrite.Port = Port;
12830 pEvtRec->u.IOPortStrWrite.cbValue = (uint8_t)cbValue;
12831 pEvtRec->u.IOPortStrWrite.cTransfers = cTransfers;
12832 pEvtRec->pNext = *pVCpu->iem.s.ppOtherEvtRecNext;
12833 *pVCpu->iem.s.ppOtherEvtRecNext = pEvtRec;
12834}
12835
12836
12837/**
12838 * Fakes and records an I/O port read.
12839 *
12840 * @returns VINF_SUCCESS.
12841 * @param pVCpu The cross context virtual CPU structure of the calling thread.
12842 * @param Port The I/O port.
12843 * @param pu32Value Where to store the fake value.
12844 * @param cbValue The size of the access.
12845 */
12846IEM_STATIC VBOXSTRICTRC iemVerifyFakeIOPortRead(PVMCPU pVCpu, RTIOPORT Port, uint32_t *pu32Value, size_t cbValue)
12847{
12848 PIEMVERIFYEVTREC pEvtRec = iemVerifyAllocRecord(pVCpu);
12849 if (pEvtRec)
12850 {
12851 pEvtRec->enmEvent = IEMVERIFYEVENT_IOPORT_READ;
12852 pEvtRec->u.IOPortRead.Port = Port;
12853 pEvtRec->u.IOPortRead.cbValue = (uint8_t)cbValue;
12854 pEvtRec->pNext = *pVCpu->iem.s.ppIemEvtRecNext;
12855 *pVCpu->iem.s.ppIemEvtRecNext = pEvtRec;
12856 }
12857 pVCpu->iem.s.cIOReads++;
12858 *pu32Value = 0xcccccccc;
12859 return VINF_SUCCESS;
12860}
12861
12862
12863/**
12864 * Fakes and records an I/O port write.
12865 *
12866 * @returns VINF_SUCCESS.
12867 * @param pVCpu The cross context virtual CPU structure of the calling thread.
12868 * @param Port The I/O port.
12869 * @param u32Value The value being written.
12870 * @param cbValue The size of the access.
12871 */
12872IEM_STATIC VBOXSTRICTRC iemVerifyFakeIOPortWrite(PVMCPU pVCpu, RTIOPORT Port, uint32_t u32Value, size_t cbValue)
12873{
12874 PIEMVERIFYEVTREC pEvtRec = iemVerifyAllocRecord(pVCpu);
12875 if (pEvtRec)
12876 {
12877 pEvtRec->enmEvent = IEMVERIFYEVENT_IOPORT_WRITE;
12878 pEvtRec->u.IOPortWrite.Port = Port;
12879 pEvtRec->u.IOPortWrite.cbValue = (uint8_t)cbValue;
12880 pEvtRec->u.IOPortWrite.u32Value = u32Value;
12881 pEvtRec->pNext = *pVCpu->iem.s.ppIemEvtRecNext;
12882 *pVCpu->iem.s.ppIemEvtRecNext = pEvtRec;
12883 }
12884 pVCpu->iem.s.cIOWrites++;
12885 return VINF_SUCCESS;
12886}
12887
12888
12889/**
12890 * Used to add extra details about a stub case.
12891 * @param pVCpu The cross context virtual CPU structure of the calling thread.
12892 */
12893IEM_STATIC void iemVerifyAssertMsg2(PVMCPU pVCpu)
12894{
12895 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
12896 PVM pVM = pVCpu->CTX_SUFF(pVM);
12897 PVMCPU pVCpu = pVCpu;
12898 char szRegs[4096];
12899 DBGFR3RegPrintf(pVM->pUVM, pVCpu->idCpu, &szRegs[0], sizeof(szRegs),
12900 "rax=%016VR{rax} rbx=%016VR{rbx} rcx=%016VR{rcx} rdx=%016VR{rdx}\n"
12901 "rsi=%016VR{rsi} rdi=%016VR{rdi} r8 =%016VR{r8} r9 =%016VR{r9}\n"
12902 "r10=%016VR{r10} r11=%016VR{r11} r12=%016VR{r12} r13=%016VR{r13}\n"
12903 "r14=%016VR{r14} r15=%016VR{r15} %VRF{rflags}\n"
12904 "rip=%016VR{rip} rsp=%016VR{rsp} rbp=%016VR{rbp}\n"
12905 "cs={%04VR{cs} base=%016VR{cs_base} limit=%08VR{cs_lim} flags=%04VR{cs_attr}} cr0=%016VR{cr0}\n"
12906 "ds={%04VR{ds} base=%016VR{ds_base} limit=%08VR{ds_lim} flags=%04VR{ds_attr}} cr2=%016VR{cr2}\n"
12907 "es={%04VR{es} base=%016VR{es_base} limit=%08VR{es_lim} flags=%04VR{es_attr}} cr3=%016VR{cr3}\n"
12908 "fs={%04VR{fs} base=%016VR{fs_base} limit=%08VR{fs_lim} flags=%04VR{fs_attr}} cr4=%016VR{cr4}\n"
12909 "gs={%04VR{gs} base=%016VR{gs_base} limit=%08VR{gs_lim} flags=%04VR{gs_attr}} cr8=%016VR{cr8}\n"
12910 "ss={%04VR{ss} base=%016VR{ss_base} limit=%08VR{ss_lim} flags=%04VR{ss_attr}}\n"
12911 "dr0=%016VR{dr0} dr1=%016VR{dr1} dr2=%016VR{dr2} dr3=%016VR{dr3}\n"
12912 "dr6=%016VR{dr6} dr7=%016VR{dr7}\n"
12913 "gdtr=%016VR{gdtr_base}:%04VR{gdtr_lim} idtr=%016VR{idtr_base}:%04VR{idtr_lim} rflags=%08VR{rflags}\n"
12914 "ldtr={%04VR{ldtr} base=%016VR{ldtr_base} limit=%08VR{ldtr_lim} flags=%08VR{ldtr_attr}}\n"
12915 "tr ={%04VR{tr} base=%016VR{tr_base} limit=%08VR{tr_lim} flags=%08VR{tr_attr}}\n"
12916 " sysenter={cs=%04VR{sysenter_cs} eip=%08VR{sysenter_eip} esp=%08VR{sysenter_esp}}\n"
12917 " efer=%016VR{efer}\n"
12918 " pat=%016VR{pat}\n"
12919 " sf_mask=%016VR{sf_mask}\n"
12920 "krnl_gs_base=%016VR{krnl_gs_base}\n"
12921 " lstar=%016VR{lstar}\n"
12922 " star=%016VR{star} cstar=%016VR{cstar}\n"
12923 "fcw=%04VR{fcw} fsw=%04VR{fsw} ftw=%04VR{ftw} mxcsr=%04VR{mxcsr} mxcsr_mask=%04VR{mxcsr_mask}\n"
12924 );
12925
12926 char szInstr1[256];
12927 DBGFR3DisasInstrEx(pVM->pUVM, pVCpu->idCpu, pVCpu->iem.s.uOldCs, pVCpu->iem.s.uOldRip,
12928 DBGF_DISAS_FLAGS_DEFAULT_MODE,
12929 szInstr1, sizeof(szInstr1), NULL);
12930 char szInstr2[256];
12931 DBGFR3DisasInstrEx(pVM->pUVM, pVCpu->idCpu, 0, 0,
12932 DBGF_DISAS_FLAGS_CURRENT_GUEST | DBGF_DISAS_FLAGS_DEFAULT_MODE,
12933 szInstr2, sizeof(szInstr2), NULL);
12934
12935 RTAssertMsg2Weak("%s%s\n%s\n", szRegs, szInstr1, szInstr2);
12936}
12937
12938
12939/**
12940 * Used by iemVerifyAssertRecord and iemVerifyAssertRecords to add a record
12941 * dump to the assertion info.
12942 *
12943 * @param pEvtRec The record to dump.
12944 */
12945IEM_STATIC void iemVerifyAssertAddRecordDump(PIEMVERIFYEVTREC pEvtRec)
12946{
12947 switch (pEvtRec->enmEvent)
12948 {
12949 case IEMVERIFYEVENT_IOPORT_READ:
12950 RTAssertMsg2Add("I/O PORT READ from %#6x, %d bytes\n",
12951 pEvtRec->u.IOPortWrite.Port,
12952 pEvtRec->u.IOPortWrite.cbValue);
12953 break;
12954 case IEMVERIFYEVENT_IOPORT_WRITE:
12955 RTAssertMsg2Add("I/O PORT WRITE to %#6x, %d bytes, value %#x\n",
12956 pEvtRec->u.IOPortWrite.Port,
12957 pEvtRec->u.IOPortWrite.cbValue,
12958 pEvtRec->u.IOPortWrite.u32Value);
12959 break;
12960 case IEMVERIFYEVENT_IOPORT_STR_READ:
12961 RTAssertMsg2Add("I/O PORT STRING READ from %#6x, %d bytes, %#x times\n",
12962 pEvtRec->u.IOPortStrWrite.Port,
12963 pEvtRec->u.IOPortStrWrite.cbValue,
12964 pEvtRec->u.IOPortStrWrite.cTransfers);
12965 break;
12966 case IEMVERIFYEVENT_IOPORT_STR_WRITE:
12967 RTAssertMsg2Add("I/O PORT STRING WRITE to %#6x, %d bytes, %#x times\n",
12968 pEvtRec->u.IOPortStrWrite.Port,
12969 pEvtRec->u.IOPortStrWrite.cbValue,
12970 pEvtRec->u.IOPortStrWrite.cTransfers);
12971 break;
12972 case IEMVERIFYEVENT_RAM_READ:
12973 RTAssertMsg2Add("RAM READ at %RGp, %#4zx bytes\n",
12974 pEvtRec->u.RamRead.GCPhys,
12975 pEvtRec->u.RamRead.cb);
12976 break;
12977 case IEMVERIFYEVENT_RAM_WRITE:
12978 RTAssertMsg2Add("RAM WRITE at %RGp, %#4zx bytes: %.*Rhxs\n",
12979 pEvtRec->u.RamWrite.GCPhys,
12980 pEvtRec->u.RamWrite.cb,
12981 (int)pEvtRec->u.RamWrite.cb,
12982 pEvtRec->u.RamWrite.ab);
12983 break;
12984 default:
12985 AssertMsgFailed(("Invalid event type %d\n", pEvtRec->enmEvent));
12986 break;
12987 }
12988}
12989
12990
12991/**
12992 * Raises an assertion on the specified record, showing the given message with
12993 * a record dump attached.
12994 *
12995 * @param pVCpu The cross context virtual CPU structure of the calling thread.
12996 * @param pEvtRec1 The first record.
12997 * @param pEvtRec2 The second record.
12998 * @param pszMsg The message explaining why we're asserting.
12999 */
13000IEM_STATIC void iemVerifyAssertRecords(PVMCPU pVCpu, PIEMVERIFYEVTREC pEvtRec1, PIEMVERIFYEVTREC pEvtRec2, const char *pszMsg)
13001{
13002 RTAssertMsg1(pszMsg, __LINE__, __FILE__, __PRETTY_FUNCTION__);
13003 iemVerifyAssertAddRecordDump(pEvtRec1);
13004 iemVerifyAssertAddRecordDump(pEvtRec2);
13005 iemVerifyAssertMsg2(pVCpu);
13006 RTAssertPanic();
13007}
13008
13009
13010/**
13011 * Raises an assertion on the specified record, showing the given message with
13012 * a record dump attached.
13013 *
13014 * @param pVCpu The cross context virtual CPU structure of the calling thread.
13015 * @param pEvtRec1 The first record.
13016 * @param pszMsg The message explaining why we're asserting.
13017 */
13018IEM_STATIC void iemVerifyAssertRecord(PVMCPU pVCpu, PIEMVERIFYEVTREC pEvtRec, const char *pszMsg)
13019{
13020 RTAssertMsg1(pszMsg, __LINE__, __FILE__, __PRETTY_FUNCTION__);
13021 iemVerifyAssertAddRecordDump(pEvtRec);
13022 iemVerifyAssertMsg2(pVCpu);
13023 RTAssertPanic();
13024}
13025
13026
13027/**
13028 * Verifies a write record.
13029 *
13030 * @param pVCpu The cross context virtual CPU structure of the calling thread.
13031 * @param pEvtRec The write record.
13032 * @param fRem Set if REM was doing the other executing. If clear
13033 * it was HM.
13034 */
13035IEM_STATIC void iemVerifyWriteRecord(PVMCPU pVCpu, PIEMVERIFYEVTREC pEvtRec, bool fRem)
13036{
13037 uint8_t abBuf[sizeof(pEvtRec->u.RamWrite.ab)]; RT_ZERO(abBuf);
13038 Assert(sizeof(abBuf) >= pEvtRec->u.RamWrite.cb);
13039 int rc = PGMPhysSimpleReadGCPhys(pVCpu->CTX_SUFF(pVM), abBuf, pEvtRec->u.RamWrite.GCPhys, pEvtRec->u.RamWrite.cb);
13040 if ( RT_FAILURE(rc)
13041 || memcmp(abBuf, pEvtRec->u.RamWrite.ab, pEvtRec->u.RamWrite.cb) )
13042 {
13043 /* fend off ins */
13044 if ( !pVCpu->iem.s.cIOReads
13045 || pEvtRec->u.RamWrite.ab[0] != 0xcc
13046 || ( pEvtRec->u.RamWrite.cb != 1
13047 && pEvtRec->u.RamWrite.cb != 2
13048 && pEvtRec->u.RamWrite.cb != 4) )
13049 {
13050 /* fend off ROMs and MMIO */
13051 if ( pEvtRec->u.RamWrite.GCPhys - UINT32_C(0x000a0000) > UINT32_C(0x60000)
13052 && pEvtRec->u.RamWrite.GCPhys - UINT32_C(0xfffc0000) > UINT32_C(0x40000) )
13053 {
13054 /* fend off fxsave */
13055 if (pEvtRec->u.RamWrite.cb != 512)
13056 {
13057 const char *pszWho = fRem ? "rem" : HMR3IsVmxEnabled(pVCpu->CTX_SUFF(pVM)->pUVM) ? "vmx" : "svm";
13058 RTAssertMsg1(NULL, __LINE__, __FILE__, __PRETTY_FUNCTION__);
13059 RTAssertMsg2Weak("Memory at %RGv differs\n", pEvtRec->u.RamWrite.GCPhys);
13060 RTAssertMsg2Add("%s: %.*Rhxs\n"
13061 "iem: %.*Rhxs\n",
13062 pszWho, pEvtRec->u.RamWrite.cb, abBuf,
13063 pEvtRec->u.RamWrite.cb, pEvtRec->u.RamWrite.ab);
13064 iemVerifyAssertAddRecordDump(pEvtRec);
13065 iemVerifyAssertMsg2(pVCpu);
13066 RTAssertPanic();
13067 }
13068 }
13069 }
13070 }
13071
13072}
13073
13074/**
13075 * Performs the post-execution verfication checks.
13076 */
13077IEM_STATIC VBOXSTRICTRC iemExecVerificationModeCheck(PVMCPU pVCpu, VBOXSTRICTRC rcStrictIem)
13078{
13079 if (!IEM_VERIFICATION_ENABLED(pVCpu))
13080 return rcStrictIem;
13081
13082 /*
13083 * Switch back the state.
13084 */
13085 PCPUMCTX pOrgCtx = CPUMQueryGuestCtxPtr(pVCpu);
13086 PCPUMCTX pDebugCtx = IEM_GET_CTX(pVCpu);
13087 Assert(pOrgCtx != pDebugCtx);
13088 IEM_GET_CTX(pVCpu) = pOrgCtx;
13089
13090 /*
13091 * Execute the instruction in REM.
13092 */
13093 bool fRem = false;
13094 PVM pVM = pVCpu->CTX_SUFF(pVM);
13095 PVMCPU pVCpu = pVCpu;
13096 VBOXSTRICTRC rc = VERR_EM_CANNOT_EXEC_GUEST;
13097#ifdef IEM_VERIFICATION_MODE_FULL_HM
13098 if ( HMIsEnabled(pVM)
13099 && pVCpu->iem.s.cIOReads == 0
13100 && pVCpu->iem.s.cIOWrites == 0
13101 && !pVCpu->iem.s.fProblematicMemory)
13102 {
13103 uint64_t uStartRip = pOrgCtx->rip;
13104 unsigned iLoops = 0;
13105 do
13106 {
13107 rc = EMR3HmSingleInstruction(pVM, pVCpu, EM_ONE_INS_FLAGS_RIP_CHANGE);
13108 iLoops++;
13109 } while ( rc == VINF_SUCCESS
13110 || ( rc == VINF_EM_DBG_STEPPED
13111 && VMCPU_FF_IS_SET(pVCpu, VMCPU_FF_INHIBIT_INTERRUPTS)
13112 && EMGetInhibitInterruptsPC(pVCpu) == pOrgCtx->rip)
13113 || ( pOrgCtx->rip != pDebugCtx->rip
13114 && pVCpu->iem.s.uInjectCpl != UINT8_MAX
13115 && iLoops < 8) );
13116 if (rc == VINF_EM_RESCHEDULE && pOrgCtx->rip != uStartRip)
13117 rc = VINF_SUCCESS;
13118 }
13119#endif
13120 if ( rc == VERR_EM_CANNOT_EXEC_GUEST
13121 || rc == VINF_IOM_R3_IOPORT_READ
13122 || rc == VINF_IOM_R3_IOPORT_WRITE
13123 || rc == VINF_IOM_R3_MMIO_READ
13124 || rc == VINF_IOM_R3_MMIO_READ_WRITE
13125 || rc == VINF_IOM_R3_MMIO_WRITE
13126 || rc == VINF_CPUM_R3_MSR_READ
13127 || rc == VINF_CPUM_R3_MSR_WRITE
13128 || rc == VINF_EM_RESCHEDULE
13129 )
13130 {
13131 EMRemLock(pVM);
13132 rc = REMR3EmulateInstruction(pVM, pVCpu);
13133 AssertRC(rc);
13134 EMRemUnlock(pVM);
13135 fRem = true;
13136 }
13137
13138# if 1 /* Skip unimplemented instructions for now. */
13139 if (rcStrictIem == VERR_IEM_INSTR_NOT_IMPLEMENTED)
13140 {
13141 IEM_GET_CTX(pVCpu) = pOrgCtx;
13142 if (rc == VINF_EM_DBG_STEPPED)
13143 return VINF_SUCCESS;
13144 return rc;
13145 }
13146# endif
13147
13148 /*
13149 * Compare the register states.
13150 */
13151 unsigned cDiffs = 0;
13152 if (memcmp(pOrgCtx, pDebugCtx, sizeof(*pDebugCtx)))
13153 {
13154 //Log(("REM and IEM ends up with different registers!\n"));
13155 const char *pszWho = fRem ? "rem" : HMR3IsVmxEnabled(pVM->pUVM) ? "vmx" : "svm";
13156
13157# define CHECK_FIELD(a_Field) \
13158 do \
13159 { \
13160 if (pOrgCtx->a_Field != pDebugCtx->a_Field) \
13161 { \
13162 switch (sizeof(pOrgCtx->a_Field)) \
13163 { \
13164 case 1: RTAssertMsg2Weak(" %8s differs - iem=%02x - %s=%02x\n", #a_Field, pDebugCtx->a_Field, pszWho, pOrgCtx->a_Field); break; \
13165 case 2: RTAssertMsg2Weak(" %8s differs - iem=%04x - %s=%04x\n", #a_Field, pDebugCtx->a_Field, pszWho, pOrgCtx->a_Field); break; \
13166 case 4: RTAssertMsg2Weak(" %8s differs - iem=%08x - %s=%08x\n", #a_Field, pDebugCtx->a_Field, pszWho, pOrgCtx->a_Field); break; \
13167 case 8: RTAssertMsg2Weak(" %8s differs - iem=%016llx - %s=%016llx\n", #a_Field, pDebugCtx->a_Field, pszWho, pOrgCtx->a_Field); break; \
13168 default: RTAssertMsg2Weak(" %8s differs\n", #a_Field); break; \
13169 } \
13170 cDiffs++; \
13171 } \
13172 } while (0)
13173# define CHECK_XSTATE_FIELD(a_Field) \
13174 do \
13175 { \
13176 if (pOrgXState->a_Field != pDebugXState->a_Field) \
13177 { \
13178 switch (sizeof(pOrgXState->a_Field)) \
13179 { \
13180 case 1: RTAssertMsg2Weak(" %8s differs - iem=%02x - %s=%02x\n", #a_Field, pDebugXState->a_Field, pszWho, pOrgXState->a_Field); break; \
13181 case 2: RTAssertMsg2Weak(" %8s differs - iem=%04x - %s=%04x\n", #a_Field, pDebugXState->a_Field, pszWho, pOrgXState->a_Field); break; \
13182 case 4: RTAssertMsg2Weak(" %8s differs - iem=%08x - %s=%08x\n", #a_Field, pDebugXState->a_Field, pszWho, pOrgXState->a_Field); break; \
13183 case 8: RTAssertMsg2Weak(" %8s differs - iem=%016llx - %s=%016llx\n", #a_Field, pDebugXState->a_Field, pszWho, pOrgXState->a_Field); break; \
13184 default: RTAssertMsg2Weak(" %8s differs\n", #a_Field); break; \
13185 } \
13186 cDiffs++; \
13187 } \
13188 } while (0)
13189
13190# define CHECK_BIT_FIELD(a_Field) \
13191 do \
13192 { \
13193 if (pOrgCtx->a_Field != pDebugCtx->a_Field) \
13194 { \
13195 RTAssertMsg2Weak(" %8s differs - iem=%02x - %s=%02x\n", #a_Field, pDebugCtx->a_Field, pszWho, pOrgCtx->a_Field); \
13196 cDiffs++; \
13197 } \
13198 } while (0)
13199
13200# define CHECK_SEL(a_Sel) \
13201 do \
13202 { \
13203 CHECK_FIELD(a_Sel.Sel); \
13204 CHECK_FIELD(a_Sel.Attr.u); \
13205 CHECK_FIELD(a_Sel.u64Base); \
13206 CHECK_FIELD(a_Sel.u32Limit); \
13207 CHECK_FIELD(a_Sel.fFlags); \
13208 } while (0)
13209
13210 PX86XSAVEAREA pOrgXState = pOrgCtx->CTX_SUFF(pXState);
13211 PX86XSAVEAREA pDebugXState = pDebugCtx->CTX_SUFF(pXState);
13212
13213#if 1 /* The recompiler doesn't update these the intel way. */
13214 if (fRem)
13215 {
13216 pOrgXState->x87.FOP = pDebugXState->x87.FOP;
13217 pOrgXState->x87.FPUIP = pDebugXState->x87.FPUIP;
13218 pOrgXState->x87.CS = pDebugXState->x87.CS;
13219 pOrgXState->x87.Rsrvd1 = pDebugXState->x87.Rsrvd1;
13220 pOrgXState->x87.FPUDP = pDebugXState->x87.FPUDP;
13221 pOrgXState->x87.DS = pDebugXState->x87.DS;
13222 pOrgXState->x87.Rsrvd2 = pDebugXState->x87.Rsrvd2;
13223 //pOrgXState->x87.MXCSR_MASK = pDebugXState->x87.MXCSR_MASK;
13224 if ((pOrgXState->x87.FSW & X86_FSW_TOP_MASK) == (pDebugXState->x87.FSW & X86_FSW_TOP_MASK))
13225 pOrgXState->x87.FSW = pDebugXState->x87.FSW;
13226 }
13227#endif
13228 if (memcmp(&pOrgXState->x87, &pDebugXState->x87, sizeof(pDebugXState->x87)))
13229 {
13230 RTAssertMsg2Weak(" the FPU state differs\n");
13231 cDiffs++;
13232 CHECK_XSTATE_FIELD(x87.FCW);
13233 CHECK_XSTATE_FIELD(x87.FSW);
13234 CHECK_XSTATE_FIELD(x87.FTW);
13235 CHECK_XSTATE_FIELD(x87.FOP);
13236 CHECK_XSTATE_FIELD(x87.FPUIP);
13237 CHECK_XSTATE_FIELD(x87.CS);
13238 CHECK_XSTATE_FIELD(x87.Rsrvd1);
13239 CHECK_XSTATE_FIELD(x87.FPUDP);
13240 CHECK_XSTATE_FIELD(x87.DS);
13241 CHECK_XSTATE_FIELD(x87.Rsrvd2);
13242 CHECK_XSTATE_FIELD(x87.MXCSR);
13243 CHECK_XSTATE_FIELD(x87.MXCSR_MASK);
13244 CHECK_XSTATE_FIELD(x87.aRegs[0].au64[0]); CHECK_XSTATE_FIELD(x87.aRegs[0].au64[1]);
13245 CHECK_XSTATE_FIELD(x87.aRegs[1].au64[0]); CHECK_XSTATE_FIELD(x87.aRegs[1].au64[1]);
13246 CHECK_XSTATE_FIELD(x87.aRegs[2].au64[0]); CHECK_XSTATE_FIELD(x87.aRegs[2].au64[1]);
13247 CHECK_XSTATE_FIELD(x87.aRegs[3].au64[0]); CHECK_XSTATE_FIELD(x87.aRegs[3].au64[1]);
13248 CHECK_XSTATE_FIELD(x87.aRegs[4].au64[0]); CHECK_XSTATE_FIELD(x87.aRegs[4].au64[1]);
13249 CHECK_XSTATE_FIELD(x87.aRegs[5].au64[0]); CHECK_XSTATE_FIELD(x87.aRegs[5].au64[1]);
13250 CHECK_XSTATE_FIELD(x87.aRegs[6].au64[0]); CHECK_XSTATE_FIELD(x87.aRegs[6].au64[1]);
13251 CHECK_XSTATE_FIELD(x87.aRegs[7].au64[0]); CHECK_XSTATE_FIELD(x87.aRegs[7].au64[1]);
13252 CHECK_XSTATE_FIELD(x87.aXMM[ 0].au64[0]); CHECK_XSTATE_FIELD(x87.aXMM[ 0].au64[1]);
13253 CHECK_XSTATE_FIELD(x87.aXMM[ 1].au64[0]); CHECK_XSTATE_FIELD(x87.aXMM[ 1].au64[1]);
13254 CHECK_XSTATE_FIELD(x87.aXMM[ 2].au64[0]); CHECK_XSTATE_FIELD(x87.aXMM[ 2].au64[1]);
13255 CHECK_XSTATE_FIELD(x87.aXMM[ 3].au64[0]); CHECK_XSTATE_FIELD(x87.aXMM[ 3].au64[1]);
13256 CHECK_XSTATE_FIELD(x87.aXMM[ 4].au64[0]); CHECK_XSTATE_FIELD(x87.aXMM[ 4].au64[1]);
13257 CHECK_XSTATE_FIELD(x87.aXMM[ 5].au64[0]); CHECK_XSTATE_FIELD(x87.aXMM[ 5].au64[1]);
13258 CHECK_XSTATE_FIELD(x87.aXMM[ 6].au64[0]); CHECK_XSTATE_FIELD(x87.aXMM[ 6].au64[1]);
13259 CHECK_XSTATE_FIELD(x87.aXMM[ 7].au64[0]); CHECK_XSTATE_FIELD(x87.aXMM[ 7].au64[1]);
13260 CHECK_XSTATE_FIELD(x87.aXMM[ 8].au64[0]); CHECK_XSTATE_FIELD(x87.aXMM[ 8].au64[1]);
13261 CHECK_XSTATE_FIELD(x87.aXMM[ 9].au64[0]); CHECK_XSTATE_FIELD(x87.aXMM[ 9].au64[1]);
13262 CHECK_XSTATE_FIELD(x87.aXMM[10].au64[0]); CHECK_XSTATE_FIELD(x87.aXMM[10].au64[1]);
13263 CHECK_XSTATE_FIELD(x87.aXMM[11].au64[0]); CHECK_XSTATE_FIELD(x87.aXMM[11].au64[1]);
13264 CHECK_XSTATE_FIELD(x87.aXMM[12].au64[0]); CHECK_XSTATE_FIELD(x87.aXMM[12].au64[1]);
13265 CHECK_XSTATE_FIELD(x87.aXMM[13].au64[0]); CHECK_XSTATE_FIELD(x87.aXMM[13].au64[1]);
13266 CHECK_XSTATE_FIELD(x87.aXMM[14].au64[0]); CHECK_XSTATE_FIELD(x87.aXMM[14].au64[1]);
13267 CHECK_XSTATE_FIELD(x87.aXMM[15].au64[0]); CHECK_XSTATE_FIELD(x87.aXMM[15].au64[1]);
13268 for (unsigned i = 0; i < RT_ELEMENTS(pOrgXState->x87.au32RsrvdRest); i++)
13269 CHECK_XSTATE_FIELD(x87.au32RsrvdRest[i]);
13270 }
13271 CHECK_FIELD(rip);
13272 uint32_t fFlagsMask = UINT32_MAX & ~pVCpu->iem.s.fUndefinedEFlags;
13273 if ((pOrgCtx->rflags.u & fFlagsMask) != (pDebugCtx->rflags.u & fFlagsMask))
13274 {
13275 RTAssertMsg2Weak(" rflags differs - iem=%08llx %s=%08llx\n", pDebugCtx->rflags.u, pszWho, pOrgCtx->rflags.u);
13276 CHECK_BIT_FIELD(rflags.Bits.u1CF);
13277 CHECK_BIT_FIELD(rflags.Bits.u1Reserved0);
13278 CHECK_BIT_FIELD(rflags.Bits.u1PF);
13279 CHECK_BIT_FIELD(rflags.Bits.u1Reserved1);
13280 CHECK_BIT_FIELD(rflags.Bits.u1AF);
13281 CHECK_BIT_FIELD(rflags.Bits.u1Reserved2);
13282 CHECK_BIT_FIELD(rflags.Bits.u1ZF);
13283 CHECK_BIT_FIELD(rflags.Bits.u1SF);
13284 CHECK_BIT_FIELD(rflags.Bits.u1TF);
13285 CHECK_BIT_FIELD(rflags.Bits.u1IF);
13286 CHECK_BIT_FIELD(rflags.Bits.u1DF);
13287 CHECK_BIT_FIELD(rflags.Bits.u1OF);
13288 CHECK_BIT_FIELD(rflags.Bits.u2IOPL);
13289 CHECK_BIT_FIELD(rflags.Bits.u1NT);
13290 CHECK_BIT_FIELD(rflags.Bits.u1Reserved3);
13291 if (0 && !fRem) /** @todo debug the occational clear RF flags when running against VT-x. */
13292 CHECK_BIT_FIELD(rflags.Bits.u1RF);
13293 CHECK_BIT_FIELD(rflags.Bits.u1VM);
13294 CHECK_BIT_FIELD(rflags.Bits.u1AC);
13295 CHECK_BIT_FIELD(rflags.Bits.u1VIF);
13296 CHECK_BIT_FIELD(rflags.Bits.u1VIP);
13297 CHECK_BIT_FIELD(rflags.Bits.u1ID);
13298 }
13299
13300 if (pVCpu->iem.s.cIOReads != 1 && !pVCpu->iem.s.fIgnoreRaxRdx)
13301 CHECK_FIELD(rax);
13302 CHECK_FIELD(rcx);
13303 if (!pVCpu->iem.s.fIgnoreRaxRdx)
13304 CHECK_FIELD(rdx);
13305 CHECK_FIELD(rbx);
13306 CHECK_FIELD(rsp);
13307 CHECK_FIELD(rbp);
13308 CHECK_FIELD(rsi);
13309 CHECK_FIELD(rdi);
13310 CHECK_FIELD(r8);
13311 CHECK_FIELD(r9);
13312 CHECK_FIELD(r10);
13313 CHECK_FIELD(r11);
13314 CHECK_FIELD(r12);
13315 CHECK_FIELD(r13);
13316 CHECK_SEL(cs);
13317 CHECK_SEL(ss);
13318 CHECK_SEL(ds);
13319 CHECK_SEL(es);
13320 CHECK_SEL(fs);
13321 CHECK_SEL(gs);
13322 CHECK_FIELD(cr0);
13323
13324 /* Klugde #1: REM fetches code and across the page boundrary and faults on the next page, while we execute
13325 the faulting instruction first: 001b:77f61ff3 66 8b 42 02 mov ax, word [edx+002h] (NT4SP1) */
13326 /* Kludge #2: CR2 differs slightly on cross page boundrary faults, we report the last address of the access
13327 while REM reports the address of the first byte on the page. Pending investigation as to which is correct. */
13328 if (pOrgCtx->cr2 != pDebugCtx->cr2)
13329 {
13330 if (pVCpu->iem.s.uOldCs == 0x1b && pVCpu->iem.s.uOldRip == 0x77f61ff3 && fRem)
13331 { /* ignore */ }
13332 else if ( (pOrgCtx->cr2 & ~(uint64_t)3) == (pDebugCtx->cr2 & ~(uint64_t)3)
13333 && (pOrgCtx->cr2 & PAGE_OFFSET_MASK) == 0
13334 && fRem)
13335 { /* ignore */ }
13336 else
13337 CHECK_FIELD(cr2);
13338 }
13339 CHECK_FIELD(cr3);
13340 CHECK_FIELD(cr4);
13341 CHECK_FIELD(dr[0]);
13342 CHECK_FIELD(dr[1]);
13343 CHECK_FIELD(dr[2]);
13344 CHECK_FIELD(dr[3]);
13345 CHECK_FIELD(dr[6]);
13346 if (!fRem || (pOrgCtx->dr[7] & ~X86_DR7_RA1_MASK) != (pDebugCtx->dr[7] & ~X86_DR7_RA1_MASK)) /* REM 'mov drX,greg' bug.*/
13347 CHECK_FIELD(dr[7]);
13348 CHECK_FIELD(gdtr.cbGdt);
13349 CHECK_FIELD(gdtr.pGdt);
13350 CHECK_FIELD(idtr.cbIdt);
13351 CHECK_FIELD(idtr.pIdt);
13352 CHECK_SEL(ldtr);
13353 CHECK_SEL(tr);
13354 CHECK_FIELD(SysEnter.cs);
13355 CHECK_FIELD(SysEnter.eip);
13356 CHECK_FIELD(SysEnter.esp);
13357 CHECK_FIELD(msrEFER);
13358 CHECK_FIELD(msrSTAR);
13359 CHECK_FIELD(msrPAT);
13360 CHECK_FIELD(msrLSTAR);
13361 CHECK_FIELD(msrCSTAR);
13362 CHECK_FIELD(msrSFMASK);
13363 CHECK_FIELD(msrKERNELGSBASE);
13364
13365 if (cDiffs != 0)
13366 {
13367 DBGFR3InfoEx(pVM->pUVM, pVCpu->idCpu, "cpumguest", "verbose", NULL);
13368 RTAssertMsg1(NULL, __LINE__, __FILE__, __FUNCTION__);
13369 RTAssertPanic();
13370 static bool volatile s_fEnterDebugger = true;
13371 if (s_fEnterDebugger)
13372 DBGFSTOP(pVM);
13373
13374# if 1 /* Ignore unimplemented instructions for now. */
13375 if (rcStrictIem == VERR_IEM_INSTR_NOT_IMPLEMENTED)
13376 rcStrictIem = VINF_SUCCESS;
13377# endif
13378 }
13379# undef CHECK_FIELD
13380# undef CHECK_BIT_FIELD
13381 }
13382
13383 /*
13384 * If the register state compared fine, check the verification event
13385 * records.
13386 */
13387 if (cDiffs == 0 && !pVCpu->iem.s.fOverlappingMovs)
13388 {
13389 /*
13390 * Compare verficiation event records.
13391 * - I/O port accesses should be a 1:1 match.
13392 */
13393 PIEMVERIFYEVTREC pIemRec = pVCpu->iem.s.pIemEvtRecHead;
13394 PIEMVERIFYEVTREC pOtherRec = pVCpu->iem.s.pOtherEvtRecHead;
13395 while (pIemRec && pOtherRec)
13396 {
13397 /* Since we might miss RAM writes and reads, ignore reads and check
13398 that any written memory is the same extra ones. */
13399 while ( IEMVERIFYEVENT_IS_RAM(pIemRec->enmEvent)
13400 && !IEMVERIFYEVENT_IS_RAM(pOtherRec->enmEvent)
13401 && pIemRec->pNext)
13402 {
13403 if (pIemRec->enmEvent == IEMVERIFYEVENT_RAM_WRITE)
13404 iemVerifyWriteRecord(pVCpu, pIemRec, fRem);
13405 pIemRec = pIemRec->pNext;
13406 }
13407
13408 /* Do the compare. */
13409 if (pIemRec->enmEvent != pOtherRec->enmEvent)
13410 {
13411 iemVerifyAssertRecords(pVCpu, pIemRec, pOtherRec, "Type mismatches");
13412 break;
13413 }
13414 bool fEquals;
13415 switch (pIemRec->enmEvent)
13416 {
13417 case IEMVERIFYEVENT_IOPORT_READ:
13418 fEquals = pIemRec->u.IOPortRead.Port == pOtherRec->u.IOPortRead.Port
13419 && pIemRec->u.IOPortRead.cbValue == pOtherRec->u.IOPortRead.cbValue;
13420 break;
13421 case IEMVERIFYEVENT_IOPORT_WRITE:
13422 fEquals = pIemRec->u.IOPortWrite.Port == pOtherRec->u.IOPortWrite.Port
13423 && pIemRec->u.IOPortWrite.cbValue == pOtherRec->u.IOPortWrite.cbValue
13424 && pIemRec->u.IOPortWrite.u32Value == pOtherRec->u.IOPortWrite.u32Value;
13425 break;
13426 case IEMVERIFYEVENT_IOPORT_STR_READ:
13427 fEquals = pIemRec->u.IOPortStrRead.Port == pOtherRec->u.IOPortStrRead.Port
13428 && pIemRec->u.IOPortStrRead.cbValue == pOtherRec->u.IOPortStrRead.cbValue
13429 && pIemRec->u.IOPortStrRead.cTransfers == pOtherRec->u.IOPortStrRead.cTransfers;
13430 break;
13431 case IEMVERIFYEVENT_IOPORT_STR_WRITE:
13432 fEquals = pIemRec->u.IOPortStrWrite.Port == pOtherRec->u.IOPortStrWrite.Port
13433 && pIemRec->u.IOPortStrWrite.cbValue == pOtherRec->u.IOPortStrWrite.cbValue
13434 && pIemRec->u.IOPortStrWrite.cTransfers == pOtherRec->u.IOPortStrWrite.cTransfers;
13435 break;
13436 case IEMVERIFYEVENT_RAM_READ:
13437 fEquals = pIemRec->u.RamRead.GCPhys == pOtherRec->u.RamRead.GCPhys
13438 && pIemRec->u.RamRead.cb == pOtherRec->u.RamRead.cb;
13439 break;
13440 case IEMVERIFYEVENT_RAM_WRITE:
13441 fEquals = pIemRec->u.RamWrite.GCPhys == pOtherRec->u.RamWrite.GCPhys
13442 && pIemRec->u.RamWrite.cb == pOtherRec->u.RamWrite.cb
13443 && !memcmp(pIemRec->u.RamWrite.ab, pOtherRec->u.RamWrite.ab, pIemRec->u.RamWrite.cb);
13444 break;
13445 default:
13446 fEquals = false;
13447 break;
13448 }
13449 if (!fEquals)
13450 {
13451 iemVerifyAssertRecords(pVCpu, pIemRec, pOtherRec, "Mismatch");
13452 break;
13453 }
13454
13455 /* advance */
13456 pIemRec = pIemRec->pNext;
13457 pOtherRec = pOtherRec->pNext;
13458 }
13459
13460 /* Ignore extra writes and reads. */
13461 while (pIemRec && IEMVERIFYEVENT_IS_RAM(pIemRec->enmEvent))
13462 {
13463 if (pIemRec->enmEvent == IEMVERIFYEVENT_RAM_WRITE)
13464 iemVerifyWriteRecord(pVCpu, pIemRec, fRem);
13465 pIemRec = pIemRec->pNext;
13466 }
13467 if (pIemRec != NULL)
13468 iemVerifyAssertRecord(pVCpu, pIemRec, "Extra IEM record!");
13469 else if (pOtherRec != NULL)
13470 iemVerifyAssertRecord(pVCpu, pOtherRec, "Extra Other record!");
13471 }
13472 IEM_GET_CTX(pVCpu) = pOrgCtx;
13473
13474 return rcStrictIem;
13475}
13476
13477#else /* !IEM_VERIFICATION_MODE_FULL || !IN_RING3 */
13478
13479/* stubs */
13480IEM_STATIC VBOXSTRICTRC iemVerifyFakeIOPortRead(PVMCPU pVCpu, RTIOPORT Port, uint32_t *pu32Value, size_t cbValue)
13481{
13482 NOREF(pVCpu); NOREF(Port); NOREF(pu32Value); NOREF(cbValue);
13483 return VERR_INTERNAL_ERROR;
13484}
13485
13486IEM_STATIC VBOXSTRICTRC iemVerifyFakeIOPortWrite(PVMCPU pVCpu, RTIOPORT Port, uint32_t u32Value, size_t cbValue)
13487{
13488 NOREF(pVCpu); NOREF(Port); NOREF(u32Value); NOREF(cbValue);
13489 return VERR_INTERNAL_ERROR;
13490}
13491
13492#endif /* !IEM_VERIFICATION_MODE_FULL || !IN_RING3 */
13493
13494
13495#ifdef LOG_ENABLED
13496/**
13497 * Logs the current instruction.
13498 * @param pVCpu The cross context virtual CPU structure of the calling EMT.
13499 * @param pCtx The current CPU context.
13500 * @param fSameCtx Set if we have the same context information as the VMM,
13501 * clear if we may have already executed an instruction in
13502 * our debug context. When clear, we assume IEMCPU holds
13503 * valid CPU mode info.
13504 */
13505IEM_STATIC void iemLogCurInstr(PVMCPU pVCpu, PCPUMCTX pCtx, bool fSameCtx)
13506{
13507# ifdef IN_RING3
13508 if (LogIs2Enabled())
13509 {
13510 char szInstr[256];
13511 uint32_t cbInstr = 0;
13512 if (fSameCtx)
13513 DBGFR3DisasInstrEx(pVCpu->pVMR3->pUVM, pVCpu->idCpu, 0, 0,
13514 DBGF_DISAS_FLAGS_CURRENT_GUEST | DBGF_DISAS_FLAGS_DEFAULT_MODE,
13515 szInstr, sizeof(szInstr), &cbInstr);
13516 else
13517 {
13518 uint32_t fFlags = 0;
13519 switch (pVCpu->iem.s.enmCpuMode)
13520 {
13521 case IEMMODE_64BIT: fFlags |= DBGF_DISAS_FLAGS_64BIT_MODE; break;
13522 case IEMMODE_32BIT: fFlags |= DBGF_DISAS_FLAGS_32BIT_MODE; break;
13523 case IEMMODE_16BIT:
13524 if (!(pCtx->cr0 & X86_CR0_PE) || pCtx->eflags.Bits.u1VM)
13525 fFlags |= DBGF_DISAS_FLAGS_16BIT_REAL_MODE;
13526 else
13527 fFlags |= DBGF_DISAS_FLAGS_16BIT_MODE;
13528 break;
13529 }
13530 DBGFR3DisasInstrEx(pVCpu->pVMR3->pUVM, pVCpu->idCpu, pCtx->cs.Sel, pCtx->rip, fFlags,
13531 szInstr, sizeof(szInstr), &cbInstr);
13532 }
13533
13534 PCX86FXSTATE pFpuCtx = &pCtx->CTX_SUFF(pXState)->x87;
13535 Log2(("****\n"
13536 " eax=%08x ebx=%08x ecx=%08x edx=%08x esi=%08x edi=%08x\n"
13537 " eip=%08x esp=%08x ebp=%08x iopl=%d tr=%04x\n"
13538 " cs=%04x ss=%04x ds=%04x es=%04x fs=%04x gs=%04x efl=%08x\n"
13539 " fsw=%04x fcw=%04x ftw=%02x mxcsr=%04x/%04x\n"
13540 " %s\n"
13541 ,
13542 pCtx->eax, pCtx->ebx, pCtx->ecx, pCtx->edx, pCtx->esi, pCtx->edi,
13543 pCtx->eip, pCtx->esp, pCtx->ebp, pCtx->eflags.Bits.u2IOPL, pCtx->tr.Sel,
13544 pCtx->cs.Sel, pCtx->ss.Sel, pCtx->ds.Sel, pCtx->es.Sel,
13545 pCtx->fs.Sel, pCtx->gs.Sel, pCtx->eflags.u,
13546 pFpuCtx->FSW, pFpuCtx->FCW, pFpuCtx->FTW, pFpuCtx->MXCSR, pFpuCtx->MXCSR_MASK,
13547 szInstr));
13548
13549 if (LogIs3Enabled())
13550 DBGFR3InfoEx(pVCpu->pVMR3->pUVM, pVCpu->idCpu, "cpumguest", "verbose", NULL);
13551 }
13552 else
13553# endif
13554 LogFlow(("IEMExecOne: cs:rip=%04x:%08RX64 ss:rsp=%04x:%08RX64 EFL=%06x\n",
13555 pCtx->cs.Sel, pCtx->rip, pCtx->ss.Sel, pCtx->rsp, pCtx->eflags.u));
13556 RT_NOREF_PV(pVCpu); RT_NOREF_PV(pCtx); RT_NOREF_PV(fSameCtx);
13557}
13558#endif
13559
13560
13561/**
13562 * Makes status code addjustments (pass up from I/O and access handler)
13563 * as well as maintaining statistics.
13564 *
13565 * @returns Strict VBox status code to pass up.
13566 * @param pVCpu The cross context virtual CPU structure of the calling thread.
13567 * @param rcStrict The status from executing an instruction.
13568 */
13569DECL_FORCE_INLINE(VBOXSTRICTRC) iemExecStatusCodeFiddling(PVMCPU pVCpu, VBOXSTRICTRC rcStrict)
13570{
13571 if (rcStrict != VINF_SUCCESS)
13572 {
13573 if (RT_SUCCESS(rcStrict))
13574 {
13575 AssertMsg( (rcStrict >= VINF_EM_FIRST && rcStrict <= VINF_EM_LAST)
13576 || rcStrict == VINF_IOM_R3_IOPORT_READ
13577 || rcStrict == VINF_IOM_R3_IOPORT_WRITE
13578 || rcStrict == VINF_IOM_R3_IOPORT_COMMIT_WRITE
13579 || rcStrict == VINF_IOM_R3_MMIO_READ
13580 || rcStrict == VINF_IOM_R3_MMIO_READ_WRITE
13581 || rcStrict == VINF_IOM_R3_MMIO_WRITE
13582 || rcStrict == VINF_IOM_R3_MMIO_COMMIT_WRITE
13583 || rcStrict == VINF_CPUM_R3_MSR_READ
13584 || rcStrict == VINF_CPUM_R3_MSR_WRITE
13585 || rcStrict == VINF_EM_RAW_EMULATE_INSTR
13586 || rcStrict == VINF_EM_RAW_TO_R3
13587 || rcStrict == VINF_EM_RAW_EMULATE_IO_BLOCK
13588 /* raw-mode / virt handlers only: */
13589 || rcStrict == VINF_EM_RAW_EMULATE_INSTR_GDT_FAULT
13590 || rcStrict == VINF_EM_RAW_EMULATE_INSTR_TSS_FAULT
13591 || rcStrict == VINF_EM_RAW_EMULATE_INSTR_LDT_FAULT
13592 || rcStrict == VINF_EM_RAW_EMULATE_INSTR_IDT_FAULT
13593 || rcStrict == VINF_SELM_SYNC_GDT
13594 || rcStrict == VINF_CSAM_PENDING_ACTION
13595 || rcStrict == VINF_PATM_CHECK_PATCH_PAGE
13596 , ("rcStrict=%Rrc\n", VBOXSTRICTRC_VAL(rcStrict)));
13597/** @todo adjust for VINF_EM_RAW_EMULATE_INSTR */
13598 int32_t const rcPassUp = pVCpu->iem.s.rcPassUp;
13599 if (rcPassUp == VINF_SUCCESS)
13600 pVCpu->iem.s.cRetInfStatuses++;
13601 else if ( rcPassUp < VINF_EM_FIRST
13602 || rcPassUp > VINF_EM_LAST
13603 || rcPassUp < VBOXSTRICTRC_VAL(rcStrict))
13604 {
13605 Log(("IEM: rcPassUp=%Rrc! rcStrict=%Rrc\n", rcPassUp, VBOXSTRICTRC_VAL(rcStrict)));
13606 pVCpu->iem.s.cRetPassUpStatus++;
13607 rcStrict = rcPassUp;
13608 }
13609 else
13610 {
13611 Log(("IEM: rcPassUp=%Rrc rcStrict=%Rrc!\n", rcPassUp, VBOXSTRICTRC_VAL(rcStrict)));
13612 pVCpu->iem.s.cRetInfStatuses++;
13613 }
13614 }
13615 else if (rcStrict == VERR_IEM_ASPECT_NOT_IMPLEMENTED)
13616 pVCpu->iem.s.cRetAspectNotImplemented++;
13617 else if (rcStrict == VERR_IEM_INSTR_NOT_IMPLEMENTED)
13618 pVCpu->iem.s.cRetInstrNotImplemented++;
13619#ifdef IEM_VERIFICATION_MODE_FULL
13620 else if (rcStrict == VERR_IEM_RESTART_INSTRUCTION)
13621 rcStrict = VINF_SUCCESS;
13622#endif
13623 else
13624 pVCpu->iem.s.cRetErrStatuses++;
13625 }
13626 else if (pVCpu->iem.s.rcPassUp != VINF_SUCCESS)
13627 {
13628 pVCpu->iem.s.cRetPassUpStatus++;
13629 rcStrict = pVCpu->iem.s.rcPassUp;
13630 }
13631
13632 return rcStrict;
13633}
13634
13635
13636/**
13637 * The actual code execution bits of IEMExecOne, IEMExecOneEx, and
13638 * IEMExecOneWithPrefetchedByPC.
13639 *
13640 * Similar code is found in IEMExecLots.
13641 *
13642 * @return Strict VBox status code.
13643 * @param pVCpu The cross context virtual CPU structure of the calling EMT.
13644 * @param pVCpu The cross context virtual CPU structure of the calling thread.
13645 * @param fExecuteInhibit If set, execute the instruction following CLI,
13646 * POP SS and MOV SS,GR.
13647 */
13648DECLINLINE(VBOXSTRICTRC) iemExecOneInner(PVMCPU pVCpu, bool fExecuteInhibit)
13649{
13650#ifdef IEM_WITH_SETJMP
13651 VBOXSTRICTRC rcStrict;
13652 jmp_buf JmpBuf;
13653 jmp_buf *pSavedJmpBuf = pVCpu->iem.s.CTX_SUFF(pJmpBuf);
13654 pVCpu->iem.s.CTX_SUFF(pJmpBuf) = &JmpBuf;
13655 if ((rcStrict = setjmp(JmpBuf)) == 0)
13656 {
13657 uint8_t b; IEM_OPCODE_GET_NEXT_U8(&b);
13658 rcStrict = FNIEMOP_CALL(g_apfnOneByteMap[b]);
13659 }
13660 else
13661 pVCpu->iem.s.cLongJumps++;
13662 pVCpu->iem.s.CTX_SUFF(pJmpBuf) = pSavedJmpBuf;
13663#else
13664 uint8_t b; IEM_OPCODE_GET_NEXT_U8(&b);
13665 VBOXSTRICTRC rcStrict = FNIEMOP_CALL(g_apfnOneByteMap[b]);
13666#endif
13667 if (rcStrict == VINF_SUCCESS)
13668 pVCpu->iem.s.cInstructions++;
13669 if (pVCpu->iem.s.cActiveMappings > 0)
13670 {
13671 Assert(rcStrict != VINF_SUCCESS);
13672 iemMemRollback(pVCpu);
13673 }
13674//#ifdef DEBUG
13675// AssertMsg(IEM_GET_INSTR_LEN(pVCpu) == cbInstr || rcStrict != VINF_SUCCESS, ("%u %u\n", IEM_GET_INSTR_LEN(pVCpu), cbInstr));
13676//#endif
13677
13678 /* Execute the next instruction as well if a cli, pop ss or
13679 mov ss, Gr has just completed successfully. */
13680 if ( fExecuteInhibit
13681 && rcStrict == VINF_SUCCESS
13682 && VMCPU_FF_IS_SET(pVCpu, VMCPU_FF_INHIBIT_INTERRUPTS)
13683 && EMGetInhibitInterruptsPC(pVCpu) == IEM_GET_CTX(pVCpu)->rip )
13684 {
13685 rcStrict = iemInitDecoderAndPrefetchOpcodes(pVCpu, pVCpu->iem.s.fBypassHandlers);
13686 if (rcStrict == VINF_SUCCESS)
13687 {
13688#ifdef LOG_ENABLED
13689 iemLogCurInstr(pVCpu, IEM_GET_CTX(pVCpu), false);
13690#endif
13691#ifdef IEM_WITH_SETJMP
13692 pVCpu->iem.s.CTX_SUFF(pJmpBuf) = &JmpBuf;
13693 if ((rcStrict = setjmp(JmpBuf)) == 0)
13694 {
13695 uint8_t b; IEM_OPCODE_GET_NEXT_U8(&b);
13696 rcStrict = FNIEMOP_CALL(g_apfnOneByteMap[b]);
13697 }
13698 else
13699 pVCpu->iem.s.cLongJumps++;
13700 pVCpu->iem.s.CTX_SUFF(pJmpBuf) = pSavedJmpBuf;
13701#else
13702 IEM_OPCODE_GET_NEXT_U8(&b);
13703 rcStrict = FNIEMOP_CALL(g_apfnOneByteMap[b]);
13704#endif
13705 if (rcStrict == VINF_SUCCESS)
13706 pVCpu->iem.s.cInstructions++;
13707 if (pVCpu->iem.s.cActiveMappings > 0)
13708 {
13709 Assert(rcStrict != VINF_SUCCESS);
13710 iemMemRollback(pVCpu);
13711 }
13712 }
13713 EMSetInhibitInterruptsPC(pVCpu, UINT64_C(0x7777555533331111));
13714 }
13715
13716 /*
13717 * Return value fiddling, statistics and sanity assertions.
13718 */
13719 rcStrict = iemExecStatusCodeFiddling(pVCpu, rcStrict);
13720
13721 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &IEM_GET_CTX(pVCpu)->cs));
13722 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &IEM_GET_CTX(pVCpu)->ss));
13723#if defined(IEM_VERIFICATION_MODE_FULL)
13724 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &IEM_GET_CTX(pVCpu)->es));
13725 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &IEM_GET_CTX(pVCpu)->ds));
13726 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &IEM_GET_CTX(pVCpu)->fs));
13727 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &IEM_GET_CTX(pVCpu)->gs));
13728#endif
13729 return rcStrict;
13730}
13731
13732
13733#ifdef IN_RC
13734/**
13735 * Re-enters raw-mode or ensure we return to ring-3.
13736 *
13737 * @returns rcStrict, maybe modified.
13738 * @param pVCpu The cross context virtual CPU structure of the calling thread.
13739 * @param pCtx The current CPU context.
13740 * @param rcStrict The status code returne by the interpreter.
13741 */
13742DECLINLINE(VBOXSTRICTRC) iemRCRawMaybeReenter(PVMCPU pVCpu, PCPUMCTX pCtx, VBOXSTRICTRC rcStrict)
13743{
13744 if ( !pVCpu->iem.s.fInPatchCode
13745 && ( rcStrict == VINF_SUCCESS
13746 || rcStrict == VERR_IEM_INSTR_NOT_IMPLEMENTED /* pgmPoolAccessPfHandlerFlush */
13747 || rcStrict == VERR_IEM_ASPECT_NOT_IMPLEMENTED /* ditto */ ) )
13748 {
13749 if (pCtx->eflags.Bits.u1IF || rcStrict != VINF_SUCCESS)
13750 CPUMRawEnter(pVCpu);
13751 else
13752 {
13753 Log(("iemRCRawMaybeReenter: VINF_EM_RESCHEDULE\n"));
13754 rcStrict = VINF_EM_RESCHEDULE;
13755 }
13756 }
13757 return rcStrict;
13758}
13759#endif
13760
13761
13762/**
13763 * Execute one instruction.
13764 *
13765 * @return Strict VBox status code.
13766 * @param pVCpu The cross context virtual CPU structure of the calling EMT.
13767 */
13768VMMDECL(VBOXSTRICTRC) IEMExecOne(PVMCPU pVCpu)
13769{
13770#if defined(IEM_VERIFICATION_MODE_FULL) && defined(IN_RING3)
13771 if (++pVCpu->iem.s.cVerifyDepth == 1)
13772 iemExecVerificationModeSetup(pVCpu);
13773#endif
13774#ifdef LOG_ENABLED
13775 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
13776 iemLogCurInstr(pVCpu, pCtx, true);
13777#endif
13778
13779 /*
13780 * Do the decoding and emulation.
13781 */
13782 VBOXSTRICTRC rcStrict = iemInitDecoderAndPrefetchOpcodes(pVCpu, false);
13783 if (rcStrict == VINF_SUCCESS)
13784 rcStrict = iemExecOneInner(pVCpu, true);
13785
13786#if defined(IEM_VERIFICATION_MODE_FULL) && defined(IN_RING3)
13787 /*
13788 * Assert some sanity.
13789 */
13790 if (pVCpu->iem.s.cVerifyDepth == 1)
13791 rcStrict = iemExecVerificationModeCheck(pVCpu, rcStrict);
13792 pVCpu->iem.s.cVerifyDepth--;
13793#endif
13794#ifdef IN_RC
13795 rcStrict = iemRCRawMaybeReenter(pVCpu, IEM_GET_CTX(pVCpu), rcStrict);
13796#endif
13797 if (rcStrict != VINF_SUCCESS)
13798 LogFlow(("IEMExecOne: cs:rip=%04x:%08RX64 ss:rsp=%04x:%08RX64 EFL=%06x - rcStrict=%Rrc\n",
13799 pCtx->cs.Sel, pCtx->rip, pCtx->ss.Sel, pCtx->rsp, pCtx->eflags.u, VBOXSTRICTRC_VAL(rcStrict)));
13800 return rcStrict;
13801}
13802
13803
13804VMMDECL(VBOXSTRICTRC) IEMExecOneEx(PVMCPU pVCpu, PCPUMCTXCORE pCtxCore, uint32_t *pcbWritten)
13805{
13806 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
13807 AssertReturn(CPUMCTX2CORE(pCtx) == pCtxCore, VERR_IEM_IPE_3);
13808
13809 uint32_t const cbOldWritten = pVCpu->iem.s.cbWritten;
13810 VBOXSTRICTRC rcStrict = iemInitDecoderAndPrefetchOpcodes(pVCpu, false);
13811 if (rcStrict == VINF_SUCCESS)
13812 {
13813 rcStrict = iemExecOneInner(pVCpu, true);
13814 if (pcbWritten)
13815 *pcbWritten = pVCpu->iem.s.cbWritten - cbOldWritten;
13816 }
13817
13818#ifdef IN_RC
13819 rcStrict = iemRCRawMaybeReenter(pVCpu, pCtx, rcStrict);
13820#endif
13821 return rcStrict;
13822}
13823
13824
13825VMMDECL(VBOXSTRICTRC) IEMExecOneWithPrefetchedByPC(PVMCPU pVCpu, PCPUMCTXCORE pCtxCore, uint64_t OpcodeBytesPC,
13826 const void *pvOpcodeBytes, size_t cbOpcodeBytes)
13827{
13828 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
13829 AssertReturn(CPUMCTX2CORE(pCtx) == pCtxCore, VERR_IEM_IPE_3);
13830
13831 VBOXSTRICTRC rcStrict;
13832 if ( cbOpcodeBytes
13833 && pCtx->rip == OpcodeBytesPC)
13834 {
13835 iemInitDecoder(pVCpu, false);
13836#ifdef IEM_WITH_CODE_TLB
13837 pVCpu->iem.s.uInstrBufPc = OpcodeBytesPC;
13838 pVCpu->iem.s.pbInstrBuf = (uint8_t const *)pvOpcodeBytes;
13839 pVCpu->iem.s.cbInstrBufTotal = (uint16_t)RT_MIN(X86_PAGE_SIZE, cbOpcodeBytes);
13840 pVCpu->iem.s.offCurInstrStart = 0;
13841 pVCpu->iem.s.offInstrNextByte = 0;
13842#else
13843 pVCpu->iem.s.cbOpcode = (uint8_t)RT_MIN(cbOpcodeBytes, sizeof(pVCpu->iem.s.abOpcode));
13844 memcpy(pVCpu->iem.s.abOpcode, pvOpcodeBytes, pVCpu->iem.s.cbOpcode);
13845#endif
13846 rcStrict = VINF_SUCCESS;
13847 }
13848 else
13849 rcStrict = iemInitDecoderAndPrefetchOpcodes(pVCpu, false);
13850 if (rcStrict == VINF_SUCCESS)
13851 {
13852 rcStrict = iemExecOneInner(pVCpu, true);
13853 }
13854
13855#ifdef IN_RC
13856 rcStrict = iemRCRawMaybeReenter(pVCpu, pCtx, rcStrict);
13857#endif
13858 return rcStrict;
13859}
13860
13861
13862VMMDECL(VBOXSTRICTRC) IEMExecOneBypassEx(PVMCPU pVCpu, PCPUMCTXCORE pCtxCore, uint32_t *pcbWritten)
13863{
13864 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
13865 AssertReturn(CPUMCTX2CORE(pCtx) == pCtxCore, VERR_IEM_IPE_3);
13866
13867 uint32_t const cbOldWritten = pVCpu->iem.s.cbWritten;
13868 VBOXSTRICTRC rcStrict = iemInitDecoderAndPrefetchOpcodes(pVCpu, true);
13869 if (rcStrict == VINF_SUCCESS)
13870 {
13871 rcStrict = iemExecOneInner(pVCpu, false);
13872 if (pcbWritten)
13873 *pcbWritten = pVCpu->iem.s.cbWritten - cbOldWritten;
13874 }
13875
13876#ifdef IN_RC
13877 rcStrict = iemRCRawMaybeReenter(pVCpu, pCtx, rcStrict);
13878#endif
13879 return rcStrict;
13880}
13881
13882
13883VMMDECL(VBOXSTRICTRC) IEMExecOneBypassWithPrefetchedByPC(PVMCPU pVCpu, PCPUMCTXCORE pCtxCore, uint64_t OpcodeBytesPC,
13884 const void *pvOpcodeBytes, size_t cbOpcodeBytes)
13885{
13886 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
13887 AssertReturn(CPUMCTX2CORE(pCtx) == pCtxCore, VERR_IEM_IPE_3);
13888
13889 VBOXSTRICTRC rcStrict;
13890 if ( cbOpcodeBytes
13891 && pCtx->rip == OpcodeBytesPC)
13892 {
13893 iemInitDecoder(pVCpu, true);
13894#ifdef IEM_WITH_CODE_TLB
13895 pVCpu->iem.s.uInstrBufPc = OpcodeBytesPC;
13896 pVCpu->iem.s.pbInstrBuf = (uint8_t const *)pvOpcodeBytes;
13897 pVCpu->iem.s.cbInstrBufTotal = (uint16_t)RT_MIN(X86_PAGE_SIZE, cbOpcodeBytes);
13898 pVCpu->iem.s.offCurInstrStart = 0;
13899 pVCpu->iem.s.offInstrNextByte = 0;
13900#else
13901 pVCpu->iem.s.cbOpcode = (uint8_t)RT_MIN(cbOpcodeBytes, sizeof(pVCpu->iem.s.abOpcode));
13902 memcpy(pVCpu->iem.s.abOpcode, pvOpcodeBytes, pVCpu->iem.s.cbOpcode);
13903#endif
13904 rcStrict = VINF_SUCCESS;
13905 }
13906 else
13907 rcStrict = iemInitDecoderAndPrefetchOpcodes(pVCpu, true);
13908 if (rcStrict == VINF_SUCCESS)
13909 rcStrict = iemExecOneInner(pVCpu, false);
13910
13911#ifdef IN_RC
13912 rcStrict = iemRCRawMaybeReenter(pVCpu, pCtx, rcStrict);
13913#endif
13914 return rcStrict;
13915}
13916
13917
13918/**
13919 * For debugging DISGetParamSize, may come in handy.
13920 *
13921 * @returns Strict VBox status code.
13922 * @param pVCpu The cross context virtual CPU structure of the
13923 * calling EMT.
13924 * @param pCtxCore The context core structure.
13925 * @param OpcodeBytesPC The PC of the opcode bytes.
13926 * @param pvOpcodeBytes Prefeched opcode bytes.
13927 * @param cbOpcodeBytes Number of prefetched bytes.
13928 * @param pcbWritten Where to return the number of bytes written.
13929 * Optional.
13930 */
13931VMMDECL(VBOXSTRICTRC) IEMExecOneBypassWithPrefetchedByPCWritten(PVMCPU pVCpu, PCPUMCTXCORE pCtxCore, uint64_t OpcodeBytesPC,
13932 const void *pvOpcodeBytes, size_t cbOpcodeBytes,
13933 uint32_t *pcbWritten)
13934{
13935 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
13936 AssertReturn(CPUMCTX2CORE(pCtx) == pCtxCore, VERR_IEM_IPE_3);
13937
13938 uint32_t const cbOldWritten = pVCpu->iem.s.cbWritten;
13939 VBOXSTRICTRC rcStrict;
13940 if ( cbOpcodeBytes
13941 && pCtx->rip == OpcodeBytesPC)
13942 {
13943 iemInitDecoder(pVCpu, true);
13944#ifdef IEM_WITH_CODE_TLB
13945 pVCpu->iem.s.uInstrBufPc = OpcodeBytesPC;
13946 pVCpu->iem.s.pbInstrBuf = (uint8_t const *)pvOpcodeBytes;
13947 pVCpu->iem.s.cbInstrBufTotal = (uint16_t)RT_MIN(X86_PAGE_SIZE, cbOpcodeBytes);
13948 pVCpu->iem.s.offCurInstrStart = 0;
13949 pVCpu->iem.s.offInstrNextByte = 0;
13950#else
13951 pVCpu->iem.s.cbOpcode = (uint8_t)RT_MIN(cbOpcodeBytes, sizeof(pVCpu->iem.s.abOpcode));
13952 memcpy(pVCpu->iem.s.abOpcode, pvOpcodeBytes, pVCpu->iem.s.cbOpcode);
13953#endif
13954 rcStrict = VINF_SUCCESS;
13955 }
13956 else
13957 rcStrict = iemInitDecoderAndPrefetchOpcodes(pVCpu, true);
13958 if (rcStrict == VINF_SUCCESS)
13959 {
13960 rcStrict = iemExecOneInner(pVCpu, false);
13961 if (pcbWritten)
13962 *pcbWritten = pVCpu->iem.s.cbWritten - cbOldWritten;
13963 }
13964
13965#ifdef IN_RC
13966 rcStrict = iemRCRawMaybeReenter(pVCpu, pCtx, rcStrict);
13967#endif
13968 return rcStrict;
13969}
13970
13971
13972VMMDECL(VBOXSTRICTRC) IEMExecLots(PVMCPU pVCpu, uint32_t *pcInstructions)
13973{
13974 uint32_t const cInstructionsAtStart = pVCpu->iem.s.cInstructions;
13975
13976#if defined(IEM_VERIFICATION_MODE_FULL) && defined(IN_RING3)
13977 /*
13978 * See if there is an interrupt pending in TRPM, inject it if we can.
13979 */
13980 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
13981# ifdef IEM_VERIFICATION_MODE_FULL
13982 pVCpu->iem.s.uInjectCpl = UINT8_MAX;
13983# endif
13984 if ( pCtx->eflags.Bits.u1IF
13985 && TRPMHasTrap(pVCpu)
13986 && EMGetInhibitInterruptsPC(pVCpu) != pCtx->rip)
13987 {
13988 uint8_t u8TrapNo;
13989 TRPMEVENT enmType;
13990 RTGCUINT uErrCode;
13991 RTGCPTR uCr2;
13992 int rc2 = TRPMQueryTrapAll(pVCpu, &u8TrapNo, &enmType, &uErrCode, &uCr2, NULL /* pu8InstLen */); AssertRC(rc2);
13993 IEMInjectTrap(pVCpu, u8TrapNo, enmType, (uint16_t)uErrCode, uCr2, 0 /* cbInstr */);
13994 if (!IEM_VERIFICATION_ENABLED(pVCpu))
13995 TRPMResetTrap(pVCpu);
13996 }
13997
13998 /*
13999 * Log the state.
14000 */
14001# ifdef LOG_ENABLED
14002 iemLogCurInstr(pVCpu, pCtx, true);
14003# endif
14004
14005 /*
14006 * Do the decoding and emulation.
14007 */
14008 VBOXSTRICTRC rcStrict = iemInitDecoderAndPrefetchOpcodes(pVCpu, false);
14009 if (rcStrict == VINF_SUCCESS)
14010 rcStrict = iemExecOneInner(pVCpu, true);
14011
14012 /*
14013 * Assert some sanity.
14014 */
14015 rcStrict = iemExecVerificationModeCheck(pVCpu, rcStrict);
14016
14017 /*
14018 * Log and return.
14019 */
14020 if (rcStrict != VINF_SUCCESS)
14021 LogFlow(("IEMExecLots: cs:rip=%04x:%08RX64 ss:rsp=%04x:%08RX64 EFL=%06x - rcStrict=%Rrc\n",
14022 pCtx->cs.Sel, pCtx->rip, pCtx->ss.Sel, pCtx->rsp, pCtx->eflags.u, VBOXSTRICTRC_VAL(rcStrict)));
14023 if (pcInstructions)
14024 *pcInstructions = pVCpu->iem.s.cInstructions - cInstructionsAtStart;
14025 return rcStrict;
14026
14027#else /* Not verification mode */
14028
14029 /*
14030 * See if there is an interrupt pending in TRPM, inject it if we can.
14031 */
14032 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
14033# ifdef IEM_VERIFICATION_MODE_FULL
14034 pVCpu->iem.s.uInjectCpl = UINT8_MAX;
14035# endif
14036 if ( pCtx->eflags.Bits.u1IF
14037 && TRPMHasTrap(pVCpu)
14038 && EMGetInhibitInterruptsPC(pVCpu) != pCtx->rip)
14039 {
14040 uint8_t u8TrapNo;
14041 TRPMEVENT enmType;
14042 RTGCUINT uErrCode;
14043 RTGCPTR uCr2;
14044 int rc2 = TRPMQueryTrapAll(pVCpu, &u8TrapNo, &enmType, &uErrCode, &uCr2, NULL /* pu8InstLen */); AssertRC(rc2);
14045 IEMInjectTrap(pVCpu, u8TrapNo, enmType, (uint16_t)uErrCode, uCr2, 0 /* cbInstr */);
14046 if (!IEM_VERIFICATION_ENABLED(pVCpu))
14047 TRPMResetTrap(pVCpu);
14048 }
14049
14050 /*
14051 * Initial decoder init w/ prefetch, then setup setjmp.
14052 */
14053 VBOXSTRICTRC rcStrict = iemInitDecoderAndPrefetchOpcodes(pVCpu, false);
14054 if (rcStrict == VINF_SUCCESS)
14055 {
14056# ifdef IEM_WITH_SETJMP
14057 jmp_buf JmpBuf;
14058 jmp_buf *pSavedJmpBuf = pVCpu->iem.s.CTX_SUFF(pJmpBuf);
14059 pVCpu->iem.s.CTX_SUFF(pJmpBuf) = &JmpBuf;
14060 pVCpu->iem.s.cActiveMappings = 0;
14061 if ((rcStrict = setjmp(JmpBuf)) == 0)
14062# endif
14063 {
14064 /*
14065 * The run loop. We limit ourselves to 4096 instructions right now.
14066 */
14067 PVM pVM = pVCpu->CTX_SUFF(pVM);
14068 uint32_t cInstr = 4096;
14069 for (;;)
14070 {
14071 /*
14072 * Log the state.
14073 */
14074# ifdef LOG_ENABLED
14075 iemLogCurInstr(pVCpu, pCtx, true);
14076# endif
14077
14078 /*
14079 * Do the decoding and emulation.
14080 */
14081 uint8_t b; IEM_OPCODE_GET_NEXT_U8(&b);
14082 rcStrict = FNIEMOP_CALL(g_apfnOneByteMap[b]);
14083 if (RT_LIKELY(rcStrict == VINF_SUCCESS))
14084 {
14085 Assert(pVCpu->iem.s.cActiveMappings == 0);
14086 pVCpu->iem.s.cInstructions++;
14087 if (RT_LIKELY(pVCpu->iem.s.rcPassUp == VINF_SUCCESS))
14088 {
14089 uint32_t fCpu = pVCpu->fLocalForcedActions
14090 & ( VMCPU_FF_ALL_MASK & ~( VMCPU_FF_PGM_SYNC_CR3
14091 | VMCPU_FF_PGM_SYNC_CR3_NON_GLOBAL
14092 | VMCPU_FF_TLB_FLUSH
14093# ifdef VBOX_WITH_RAW_MODE
14094 | VMCPU_FF_TRPM_SYNC_IDT
14095 | VMCPU_FF_SELM_SYNC_TSS
14096 | VMCPU_FF_SELM_SYNC_GDT
14097 | VMCPU_FF_SELM_SYNC_LDT
14098# endif
14099 | VMCPU_FF_INHIBIT_INTERRUPTS
14100 | VMCPU_FF_BLOCK_NMIS ));
14101
14102 if (RT_LIKELY( ( !fCpu
14103 || ( !(fCpu & ~(VMCPU_FF_INTERRUPT_APIC | VMCPU_FF_INTERRUPT_PIC))
14104 && !pCtx->rflags.Bits.u1IF) )
14105 && !VM_FF_IS_PENDING(pVM, VM_FF_ALL_MASK) ))
14106 {
14107 if (cInstr-- > 0)
14108 {
14109 Assert(pVCpu->iem.s.cActiveMappings == 0);
14110 iemReInitDecoder(pVCpu);
14111 continue;
14112 }
14113 }
14114 }
14115 Assert(pVCpu->iem.s.cActiveMappings == 0);
14116 }
14117 else if (pVCpu->iem.s.cActiveMappings > 0)
14118 iemMemRollback(pVCpu);
14119 rcStrict = iemExecStatusCodeFiddling(pVCpu, rcStrict);
14120 break;
14121 }
14122 }
14123# ifdef IEM_WITH_SETJMP
14124 else
14125 {
14126 if (pVCpu->iem.s.cActiveMappings > 0)
14127 iemMemRollback(pVCpu);
14128 pVCpu->iem.s.cLongJumps++;
14129 }
14130 pVCpu->iem.s.CTX_SUFF(pJmpBuf) = pSavedJmpBuf;
14131# endif
14132
14133 /*
14134 * Assert hidden register sanity (also done in iemInitDecoder and iemReInitDecoder).
14135 */
14136 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &IEM_GET_CTX(pVCpu)->cs));
14137 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &IEM_GET_CTX(pVCpu)->ss));
14138# if defined(IEM_VERIFICATION_MODE_FULL)
14139 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &IEM_GET_CTX(pVCpu)->es));
14140 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &IEM_GET_CTX(pVCpu)->ds));
14141 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &IEM_GET_CTX(pVCpu)->fs));
14142 Assert(CPUMSELREG_ARE_HIDDEN_PARTS_VALID(pVCpu, &IEM_GET_CTX(pVCpu)->gs));
14143# endif
14144 }
14145
14146 /*
14147 * Maybe re-enter raw-mode and log.
14148 */
14149# ifdef IN_RC
14150 rcStrict = iemRCRawMaybeReenter(pVCpu, IEM_GET_CTX(pVCpu), rcStrict);
14151# endif
14152 if (rcStrict != VINF_SUCCESS)
14153 LogFlow(("IEMExecLots: cs:rip=%04x:%08RX64 ss:rsp=%04x:%08RX64 EFL=%06x - rcStrict=%Rrc\n",
14154 pCtx->cs.Sel, pCtx->rip, pCtx->ss.Sel, pCtx->rsp, pCtx->eflags.u, VBOXSTRICTRC_VAL(rcStrict)));
14155 if (pcInstructions)
14156 *pcInstructions = pVCpu->iem.s.cInstructions - cInstructionsAtStart;
14157 return rcStrict;
14158#endif /* Not verification mode */
14159}
14160
14161
14162
14163/**
14164 * Injects a trap, fault, abort, software interrupt or external interrupt.
14165 *
14166 * The parameter list matches TRPMQueryTrapAll pretty closely.
14167 *
14168 * @returns Strict VBox status code.
14169 * @param pVCpu The cross context virtual CPU structure of the calling EMT.
14170 * @param u8TrapNo The trap number.
14171 * @param enmType What type is it (trap/fault/abort), software
14172 * interrupt or hardware interrupt.
14173 * @param uErrCode The error code if applicable.
14174 * @param uCr2 The CR2 value if applicable.
14175 * @param cbInstr The instruction length (only relevant for
14176 * software interrupts).
14177 */
14178VMM_INT_DECL(VBOXSTRICTRC) IEMInjectTrap(PVMCPU pVCpu, uint8_t u8TrapNo, TRPMEVENT enmType, uint16_t uErrCode, RTGCPTR uCr2,
14179 uint8_t cbInstr)
14180{
14181 iemInitDecoder(pVCpu, false);
14182#ifdef DBGFTRACE_ENABLED
14183 RTTraceBufAddMsgF(pVCpu->CTX_SUFF(pVM)->CTX_SUFF(hTraceBuf), "IEMInjectTrap: %x %d %x %llx",
14184 u8TrapNo, enmType, uErrCode, uCr2);
14185#endif
14186
14187 uint32_t fFlags;
14188 switch (enmType)
14189 {
14190 case TRPM_HARDWARE_INT:
14191 Log(("IEMInjectTrap: %#4x ext\n", u8TrapNo));
14192 fFlags = IEM_XCPT_FLAGS_T_EXT_INT;
14193 uErrCode = uCr2 = 0;
14194 break;
14195
14196 case TRPM_SOFTWARE_INT:
14197 Log(("IEMInjectTrap: %#4x soft\n", u8TrapNo));
14198 fFlags = IEM_XCPT_FLAGS_T_SOFT_INT;
14199 uErrCode = uCr2 = 0;
14200 break;
14201
14202 case TRPM_TRAP:
14203 Log(("IEMInjectTrap: %#4x trap err=%#x cr2=%#RGv\n", u8TrapNo, uErrCode, uCr2));
14204 fFlags = IEM_XCPT_FLAGS_T_CPU_XCPT;
14205 if (u8TrapNo == X86_XCPT_PF)
14206 fFlags |= IEM_XCPT_FLAGS_CR2;
14207 switch (u8TrapNo)
14208 {
14209 case X86_XCPT_DF:
14210 case X86_XCPT_TS:
14211 case X86_XCPT_NP:
14212 case X86_XCPT_SS:
14213 case X86_XCPT_PF:
14214 case X86_XCPT_AC:
14215 fFlags |= IEM_XCPT_FLAGS_ERR;
14216 break;
14217
14218 case X86_XCPT_NMI:
14219 VMCPU_FF_SET(pVCpu, VMCPU_FF_BLOCK_NMIS);
14220 break;
14221 }
14222 break;
14223
14224 IEM_NOT_REACHED_DEFAULT_CASE_RET();
14225 }
14226
14227 return iemRaiseXcptOrInt(pVCpu, cbInstr, u8TrapNo, fFlags, uErrCode, uCr2);
14228}
14229
14230
14231/**
14232 * Injects the active TRPM event.
14233 *
14234 * @returns Strict VBox status code.
14235 * @param pVCpu The cross context virtual CPU structure.
14236 */
14237VMMDECL(VBOXSTRICTRC) IEMInjectTrpmEvent(PVMCPU pVCpu)
14238{
14239#ifndef IEM_IMPLEMENTS_TASKSWITCH
14240 IEM_RETURN_ASPECT_NOT_IMPLEMENTED_LOG(("Event injection\n"));
14241#else
14242 uint8_t u8TrapNo;
14243 TRPMEVENT enmType;
14244 RTGCUINT uErrCode;
14245 RTGCUINTPTR uCr2;
14246 uint8_t cbInstr;
14247 int rc = TRPMQueryTrapAll(pVCpu, &u8TrapNo, &enmType, &uErrCode, &uCr2, &cbInstr);
14248 if (RT_FAILURE(rc))
14249 return rc;
14250
14251 VBOXSTRICTRC rcStrict = IEMInjectTrap(pVCpu, u8TrapNo, enmType, uErrCode, uCr2, cbInstr);
14252
14253 /** @todo Are there any other codes that imply the event was successfully
14254 * delivered to the guest? See @bugref{6607}. */
14255 if ( rcStrict == VINF_SUCCESS
14256 || rcStrict == VINF_IEM_RAISED_XCPT)
14257 {
14258 TRPMResetTrap(pVCpu);
14259 }
14260 return rcStrict;
14261#endif
14262}
14263
14264
14265VMM_INT_DECL(int) IEMBreakpointSet(PVM pVM, RTGCPTR GCPtrBp)
14266{
14267 RT_NOREF_PV(pVM); RT_NOREF_PV(GCPtrBp);
14268 return VERR_NOT_IMPLEMENTED;
14269}
14270
14271
14272VMM_INT_DECL(int) IEMBreakpointClear(PVM pVM, RTGCPTR GCPtrBp)
14273{
14274 RT_NOREF_PV(pVM); RT_NOREF_PV(GCPtrBp);
14275 return VERR_NOT_IMPLEMENTED;
14276}
14277
14278
14279#if 0 /* The IRET-to-v8086 mode in PATM is very optimistic, so I don't dare do this yet. */
14280/**
14281 * Executes a IRET instruction with default operand size.
14282 *
14283 * This is for PATM.
14284 *
14285 * @returns VBox status code.
14286 * @param pVCpu The cross context virtual CPU structure of the calling EMT.
14287 * @param pCtxCore The register frame.
14288 */
14289VMM_INT_DECL(int) IEMExecInstr_iret(PVMCPU pVCpu, PCPUMCTXCORE pCtxCore)
14290{
14291 PCPUMCTX pCtx = IEM_GET_CTX(pVCpu);
14292
14293 iemCtxCoreToCtx(pCtx, pCtxCore);
14294 iemInitDecoder(pVCpu);
14295 VBOXSTRICTRC rcStrict = iemCImpl_iret(pVCpu, 1, pVCpu->iem.s.enmDefOpSize);
14296 if (rcStrict == VINF_SUCCESS)
14297 iemCtxToCtxCore(pCtxCore, pCtx);
14298 else
14299 LogFlow(("IEMExecInstr_iret: cs:rip=%04x:%08RX64 ss:rsp=%04x:%08RX64 EFL=%06x - rcStrict=%Rrc\n",
14300 pCtx->cs, pCtx->rip, pCtx->ss, pCtx->rsp, pCtx->eflags.u, VBOXSTRICTRC_VAL(rcStrict)));
14301 return rcStrict;
14302}
14303#endif
14304
14305
14306/**
14307 * Macro used by the IEMExec* method to check the given instruction length.
14308 *
14309 * Will return on failure!
14310 *
14311 * @param a_cbInstr The given instruction length.
14312 * @param a_cbMin The minimum length.
14313 */
14314#define IEMEXEC_ASSERT_INSTR_LEN_RETURN(a_cbInstr, a_cbMin) \
14315 AssertMsgReturn((unsigned)(a_cbInstr) - (unsigned)(a_cbMin) <= (unsigned)15 - (unsigned)(a_cbMin), \
14316 ("cbInstr=%u cbMin=%u\n", (a_cbInstr), (a_cbMin)), VERR_IEM_INVALID_INSTR_LENGTH)
14317
14318
14319/**
14320 * Calls iemUninitExec, iemExecStatusCodeFiddling and iemRCRawMaybeReenter.
14321 *
14322 * Only calling iemRCRawMaybeReenter in raw-mode, obviously.
14323 *
14324 * @returns Fiddled strict vbox status code, ready to return to non-IEM caller.
14325 * @param pVCpu The cross context virtual CPU structure of the calling thread.
14326 * @param rcStrict The status code to fiddle.
14327 */
14328DECLINLINE(VBOXSTRICTRC) iemUninitExecAndFiddleStatusAndMaybeReenter(PVMCPU pVCpu, VBOXSTRICTRC rcStrict)
14329{
14330 iemUninitExec(pVCpu);
14331#ifdef IN_RC
14332 return iemRCRawMaybeReenter(pVCpu, IEM_GET_CTX(pVCpu),
14333 iemExecStatusCodeFiddling(pVCpu, rcStrict));
14334#else
14335 return iemExecStatusCodeFiddling(pVCpu, rcStrict);
14336#endif
14337}
14338
14339
14340/**
14341 * Interface for HM and EM for executing string I/O OUT (write) instructions.
14342 *
14343 * This API ASSUMES that the caller has already verified that the guest code is
14344 * allowed to access the I/O port. (The I/O port is in the DX register in the
14345 * guest state.)
14346 *
14347 * @returns Strict VBox status code.
14348 * @param pVCpu The cross context virtual CPU structure.
14349 * @param cbValue The size of the I/O port access (1, 2, or 4).
14350 * @param enmAddrMode The addressing mode.
14351 * @param fRepPrefix Indicates whether a repeat prefix is used
14352 * (doesn't matter which for this instruction).
14353 * @param cbInstr The instruction length in bytes.
14354 * @param iEffSeg The effective segment address.
14355 * @param fIoChecked Whether the access to the I/O port has been
14356 * checked or not. It's typically checked in the
14357 * HM scenario.
14358 */
14359VMM_INT_DECL(VBOXSTRICTRC) IEMExecStringIoWrite(PVMCPU pVCpu, uint8_t cbValue, IEMMODE enmAddrMode,
14360 bool fRepPrefix, uint8_t cbInstr, uint8_t iEffSeg, bool fIoChecked)
14361{
14362 AssertMsgReturn(iEffSeg < X86_SREG_COUNT, ("%#x\n", iEffSeg), VERR_IEM_INVALID_EFF_SEG);
14363 IEMEXEC_ASSERT_INSTR_LEN_RETURN(cbInstr, 1);
14364
14365 /*
14366 * State init.
14367 */
14368 iemInitExec(pVCpu, false /*fBypassHandlers*/);
14369
14370 /*
14371 * Switch orgy for getting to the right handler.
14372 */
14373 VBOXSTRICTRC rcStrict;
14374 if (fRepPrefix)
14375 {
14376 switch (enmAddrMode)
14377 {
14378 case IEMMODE_16BIT:
14379 switch (cbValue)
14380 {
14381 case 1: rcStrict = iemCImpl_rep_outs_op8_addr16(pVCpu, cbInstr, iEffSeg, fIoChecked); break;
14382 case 2: rcStrict = iemCImpl_rep_outs_op16_addr16(pVCpu, cbInstr, iEffSeg, fIoChecked); break;
14383 case 4: rcStrict = iemCImpl_rep_outs_op32_addr16(pVCpu, cbInstr, iEffSeg, fIoChecked); break;
14384 default:
14385 AssertMsgFailedReturn(("cbValue=%#x\n", cbValue), VERR_IEM_INVALID_OPERAND_SIZE);
14386 }
14387 break;
14388
14389 case IEMMODE_32BIT:
14390 switch (cbValue)
14391 {
14392 case 1: rcStrict = iemCImpl_rep_outs_op8_addr32(pVCpu, cbInstr, iEffSeg, fIoChecked); break;
14393 case 2: rcStrict = iemCImpl_rep_outs_op16_addr32(pVCpu, cbInstr, iEffSeg, fIoChecked); break;
14394 case 4: rcStrict = iemCImpl_rep_outs_op32_addr32(pVCpu, cbInstr, iEffSeg, fIoChecked); break;
14395 default:
14396 AssertMsgFailedReturn(("cbValue=%#x\n", cbValue), VERR_IEM_INVALID_OPERAND_SIZE);
14397 }
14398 break;
14399
14400 case IEMMODE_64BIT:
14401 switch (cbValue)
14402 {
14403 case 1: rcStrict = iemCImpl_rep_outs_op8_addr64(pVCpu, cbInstr, iEffSeg, fIoChecked); break;
14404 case 2: rcStrict = iemCImpl_rep_outs_op16_addr64(pVCpu, cbInstr, iEffSeg, fIoChecked); break;
14405 case 4: rcStrict = iemCImpl_rep_outs_op32_addr64(pVCpu, cbInstr, iEffSeg, fIoChecked); break;
14406 default:
14407 AssertMsgFailedReturn(("cbValue=%#x\n", cbValue), VERR_IEM_INVALID_OPERAND_SIZE);
14408 }
14409 break;
14410
14411 default:
14412 AssertMsgFailedReturn(("enmAddrMode=%d\n", enmAddrMode), VERR_IEM_INVALID_ADDRESS_MODE);
14413 }
14414 }
14415 else
14416 {
14417 switch (enmAddrMode)
14418 {
14419 case IEMMODE_16BIT:
14420 switch (cbValue)
14421 {
14422 case 1: rcStrict = iemCImpl_outs_op8_addr16(pVCpu, cbInstr, iEffSeg, fIoChecked); break;
14423 case 2: rcStrict = iemCImpl_outs_op16_addr16(pVCpu, cbInstr, iEffSeg, fIoChecked); break;
14424 case 4: rcStrict = iemCImpl_outs_op32_addr16(pVCpu, cbInstr, iEffSeg, fIoChecked); break;
14425 default:
14426 AssertMsgFailedReturn(("cbValue=%#x\n", cbValue), VERR_IEM_INVALID_OPERAND_SIZE);
14427 }
14428 break;
14429
14430 case IEMMODE_32BIT:
14431 switch (cbValue)
14432 {
14433 case 1: rcStrict = iemCImpl_outs_op8_addr32(pVCpu, cbInstr, iEffSeg, fIoChecked); break;
14434 case 2: rcStrict = iemCImpl_outs_op16_addr32(pVCpu, cbInstr, iEffSeg, fIoChecked); break;
14435 case 4: rcStrict = iemCImpl_outs_op32_addr32(pVCpu, cbInstr, iEffSeg, fIoChecked); break;
14436 default:
14437 AssertMsgFailedReturn(("cbValue=%#x\n", cbValue), VERR_IEM_INVALID_OPERAND_SIZE);
14438 }
14439 break;
14440
14441 case IEMMODE_64BIT:
14442 switch (cbValue)
14443 {
14444 case 1: rcStrict = iemCImpl_outs_op8_addr64(pVCpu, cbInstr, iEffSeg, fIoChecked); break;
14445 case 2: rcStrict = iemCImpl_outs_op16_addr64(pVCpu, cbInstr, iEffSeg, fIoChecked); break;
14446 case 4: rcStrict = iemCImpl_outs_op32_addr64(pVCpu, cbInstr, iEffSeg, fIoChecked); break;
14447 default:
14448 AssertMsgFailedReturn(("cbValue=%#x\n", cbValue), VERR_IEM_INVALID_OPERAND_SIZE);
14449 }
14450 break;
14451
14452 default:
14453 AssertMsgFailedReturn(("enmAddrMode=%d\n", enmAddrMode), VERR_IEM_INVALID_ADDRESS_MODE);
14454 }
14455 }
14456
14457 return iemUninitExecAndFiddleStatusAndMaybeReenter(pVCpu, rcStrict);
14458}
14459
14460
14461/**
14462 * Interface for HM and EM for executing string I/O IN (read) instructions.
14463 *
14464 * This API ASSUMES that the caller has already verified that the guest code is
14465 * allowed to access the I/O port. (The I/O port is in the DX register in the
14466 * guest state.)
14467 *
14468 * @returns Strict VBox status code.
14469 * @param pVCpu The cross context virtual CPU structure.
14470 * @param cbValue The size of the I/O port access (1, 2, or 4).
14471 * @param enmAddrMode The addressing mode.
14472 * @param fRepPrefix Indicates whether a repeat prefix is used
14473 * (doesn't matter which for this instruction).
14474 * @param cbInstr The instruction length in bytes.
14475 * @param fIoChecked Whether the access to the I/O port has been
14476 * checked or not. It's typically checked in the
14477 * HM scenario.
14478 */
14479VMM_INT_DECL(VBOXSTRICTRC) IEMExecStringIoRead(PVMCPU pVCpu, uint8_t cbValue, IEMMODE enmAddrMode,
14480 bool fRepPrefix, uint8_t cbInstr, bool fIoChecked)
14481{
14482 IEMEXEC_ASSERT_INSTR_LEN_RETURN(cbInstr, 1);
14483
14484 /*
14485 * State init.
14486 */
14487 iemInitExec(pVCpu, false /*fBypassHandlers*/);
14488
14489 /*
14490 * Switch orgy for getting to the right handler.
14491 */
14492 VBOXSTRICTRC rcStrict;
14493 if (fRepPrefix)
14494 {
14495 switch (enmAddrMode)
14496 {
14497 case IEMMODE_16BIT:
14498 switch (cbValue)
14499 {
14500 case 1: rcStrict = iemCImpl_rep_ins_op8_addr16(pVCpu, cbInstr, fIoChecked); break;
14501 case 2: rcStrict = iemCImpl_rep_ins_op16_addr16(pVCpu, cbInstr, fIoChecked); break;
14502 case 4: rcStrict = iemCImpl_rep_ins_op32_addr16(pVCpu, cbInstr, fIoChecked); break;
14503 default:
14504 AssertMsgFailedReturn(("cbValue=%#x\n", cbValue), VERR_IEM_INVALID_OPERAND_SIZE);
14505 }
14506 break;
14507
14508 case IEMMODE_32BIT:
14509 switch (cbValue)
14510 {
14511 case 1: rcStrict = iemCImpl_rep_ins_op8_addr32(pVCpu, cbInstr, fIoChecked); break;
14512 case 2: rcStrict = iemCImpl_rep_ins_op16_addr32(pVCpu, cbInstr, fIoChecked); break;
14513 case 4: rcStrict = iemCImpl_rep_ins_op32_addr32(pVCpu, cbInstr, fIoChecked); break;
14514 default:
14515 AssertMsgFailedReturn(("cbValue=%#x\n", cbValue), VERR_IEM_INVALID_OPERAND_SIZE);
14516 }
14517 break;
14518
14519 case IEMMODE_64BIT:
14520 switch (cbValue)
14521 {
14522 case 1: rcStrict = iemCImpl_rep_ins_op8_addr64(pVCpu, cbInstr, fIoChecked); break;
14523 case 2: rcStrict = iemCImpl_rep_ins_op16_addr64(pVCpu, cbInstr, fIoChecked); break;
14524 case 4: rcStrict = iemCImpl_rep_ins_op32_addr64(pVCpu, cbInstr, fIoChecked); break;
14525 default:
14526 AssertMsgFailedReturn(("cbValue=%#x\n", cbValue), VERR_IEM_INVALID_OPERAND_SIZE);
14527 }
14528 break;
14529
14530 default:
14531 AssertMsgFailedReturn(("enmAddrMode=%d\n", enmAddrMode), VERR_IEM_INVALID_ADDRESS_MODE);
14532 }
14533 }
14534 else
14535 {
14536 switch (enmAddrMode)
14537 {
14538 case IEMMODE_16BIT:
14539 switch (cbValue)
14540 {
14541 case 1: rcStrict = iemCImpl_ins_op8_addr16(pVCpu, cbInstr, fIoChecked); break;
14542 case 2: rcStrict = iemCImpl_ins_op16_addr16(pVCpu, cbInstr, fIoChecked); break;
14543 case 4: rcStrict = iemCImpl_ins_op32_addr16(pVCpu, cbInstr, fIoChecked); break;
14544 default:
14545 AssertMsgFailedReturn(("cbValue=%#x\n", cbValue), VERR_IEM_INVALID_OPERAND_SIZE);
14546 }
14547 break;
14548
14549 case IEMMODE_32BIT:
14550 switch (cbValue)
14551 {
14552 case 1: rcStrict = iemCImpl_ins_op8_addr32(pVCpu, cbInstr, fIoChecked); break;
14553 case 2: rcStrict = iemCImpl_ins_op16_addr32(pVCpu, cbInstr, fIoChecked); break;
14554 case 4: rcStrict = iemCImpl_ins_op32_addr32(pVCpu, cbInstr, fIoChecked); break;
14555 default:
14556 AssertMsgFailedReturn(("cbValue=%#x\n", cbValue), VERR_IEM_INVALID_OPERAND_SIZE);
14557 }
14558 break;
14559
14560 case IEMMODE_64BIT:
14561 switch (cbValue)
14562 {
14563 case 1: rcStrict = iemCImpl_ins_op8_addr64(pVCpu, cbInstr, fIoChecked); break;
14564 case 2: rcStrict = iemCImpl_ins_op16_addr64(pVCpu, cbInstr, fIoChecked); break;
14565 case 4: rcStrict = iemCImpl_ins_op32_addr64(pVCpu, cbInstr, fIoChecked); break;
14566 default:
14567 AssertMsgFailedReturn(("cbValue=%#x\n", cbValue), VERR_IEM_INVALID_OPERAND_SIZE);
14568 }
14569 break;
14570
14571 default:
14572 AssertMsgFailedReturn(("enmAddrMode=%d\n", enmAddrMode), VERR_IEM_INVALID_ADDRESS_MODE);
14573 }
14574 }
14575
14576 return iemUninitExecAndFiddleStatusAndMaybeReenter(pVCpu, rcStrict);
14577}
14578
14579
14580/**
14581 * Interface for rawmode to write execute an OUT instruction.
14582 *
14583 * @returns Strict VBox status code.
14584 * @param pVCpu The cross context virtual CPU structure.
14585 * @param cbInstr The instruction length in bytes.
14586 * @param u16Port The port to read.
14587 * @param cbReg The register size.
14588 *
14589 * @remarks In ring-0 not all of the state needs to be synced in.
14590 */
14591VMM_INT_DECL(VBOXSTRICTRC) IEMExecDecodedOut(PVMCPU pVCpu, uint8_t cbInstr, uint16_t u16Port, uint8_t cbReg)
14592{
14593 IEMEXEC_ASSERT_INSTR_LEN_RETURN(cbInstr, 1);
14594 Assert(cbReg <= 4 && cbReg != 3);
14595
14596 iemInitExec(pVCpu, false /*fBypassHandlers*/);
14597 VBOXSTRICTRC rcStrict = IEM_CIMPL_CALL_2(iemCImpl_out, u16Port, cbReg);
14598 return iemUninitExecAndFiddleStatusAndMaybeReenter(pVCpu, rcStrict);
14599}
14600
14601
14602/**
14603 * Interface for rawmode to write execute an IN instruction.
14604 *
14605 * @returns Strict VBox status code.
14606 * @param pVCpu The cross context virtual CPU structure.
14607 * @param cbInstr The instruction length in bytes.
14608 * @param u16Port The port to read.
14609 * @param cbReg The register size.
14610 */
14611VMM_INT_DECL(VBOXSTRICTRC) IEMExecDecodedIn(PVMCPU pVCpu, uint8_t cbInstr, uint16_t u16Port, uint8_t cbReg)
14612{
14613 IEMEXEC_ASSERT_INSTR_LEN_RETURN(cbInstr, 1);
14614 Assert(cbReg <= 4 && cbReg != 3);
14615
14616 iemInitExec(pVCpu, false /*fBypassHandlers*/);
14617 VBOXSTRICTRC rcStrict = IEM_CIMPL_CALL_2(iemCImpl_in, u16Port, cbReg);
14618 return iemUninitExecAndFiddleStatusAndMaybeReenter(pVCpu, rcStrict);
14619}
14620
14621
14622/**
14623 * Interface for HM and EM to write to a CRx register.
14624 *
14625 * @returns Strict VBox status code.
14626 * @param pVCpu The cross context virtual CPU structure.
14627 * @param cbInstr The instruction length in bytes.
14628 * @param iCrReg The control register number (destination).
14629 * @param iGReg The general purpose register number (source).
14630 *
14631 * @remarks In ring-0 not all of the state needs to be synced in.
14632 */
14633VMM_INT_DECL(VBOXSTRICTRC) IEMExecDecodedMovCRxWrite(PVMCPU pVCpu, uint8_t cbInstr, uint8_t iCrReg, uint8_t iGReg)
14634{
14635 IEMEXEC_ASSERT_INSTR_LEN_RETURN(cbInstr, 2);
14636 Assert(iCrReg < 16);
14637 Assert(iGReg < 16);
14638
14639 iemInitExec(pVCpu, false /*fBypassHandlers*/);
14640 VBOXSTRICTRC rcStrict = IEM_CIMPL_CALL_2(iemCImpl_mov_Cd_Rd, iCrReg, iGReg);
14641 return iemUninitExecAndFiddleStatusAndMaybeReenter(pVCpu, rcStrict);
14642}
14643
14644
14645/**
14646 * Interface for HM and EM to read from a CRx register.
14647 *
14648 * @returns Strict VBox status code.
14649 * @param pVCpu The cross context virtual CPU structure.
14650 * @param cbInstr The instruction length in bytes.
14651 * @param iGReg The general purpose register number (destination).
14652 * @param iCrReg The control register number (source).
14653 *
14654 * @remarks In ring-0 not all of the state needs to be synced in.
14655 */
14656VMM_INT_DECL(VBOXSTRICTRC) IEMExecDecodedMovCRxRead(PVMCPU pVCpu, uint8_t cbInstr, uint8_t iGReg, uint8_t iCrReg)
14657{
14658 IEMEXEC_ASSERT_INSTR_LEN_RETURN(cbInstr, 2);
14659 Assert(iCrReg < 16);
14660 Assert(iGReg < 16);
14661
14662 iemInitExec(pVCpu, false /*fBypassHandlers*/);
14663 VBOXSTRICTRC rcStrict = IEM_CIMPL_CALL_2(iemCImpl_mov_Rd_Cd, iGReg, iCrReg);
14664 return iemUninitExecAndFiddleStatusAndMaybeReenter(pVCpu, rcStrict);
14665}
14666
14667
14668/**
14669 * Interface for HM and EM to clear the CR0[TS] bit.
14670 *
14671 * @returns Strict VBox status code.
14672 * @param pVCpu The cross context virtual CPU structure.
14673 * @param cbInstr The instruction length in bytes.
14674 *
14675 * @remarks In ring-0 not all of the state needs to be synced in.
14676 */
14677VMM_INT_DECL(VBOXSTRICTRC) IEMExecDecodedClts(PVMCPU pVCpu, uint8_t cbInstr)
14678{
14679 IEMEXEC_ASSERT_INSTR_LEN_RETURN(cbInstr, 2);
14680
14681 iemInitExec(pVCpu, false /*fBypassHandlers*/);
14682 VBOXSTRICTRC rcStrict = IEM_CIMPL_CALL_0(iemCImpl_clts);
14683 return iemUninitExecAndFiddleStatusAndMaybeReenter(pVCpu, rcStrict);
14684}
14685
14686
14687/**
14688 * Interface for HM and EM to emulate the LMSW instruction (loads CR0).
14689 *
14690 * @returns Strict VBox status code.
14691 * @param pVCpu The cross context virtual CPU structure.
14692 * @param cbInstr The instruction length in bytes.
14693 * @param uValue The value to load into CR0.
14694 *
14695 * @remarks In ring-0 not all of the state needs to be synced in.
14696 */
14697VMM_INT_DECL(VBOXSTRICTRC) IEMExecDecodedLmsw(PVMCPU pVCpu, uint8_t cbInstr, uint16_t uValue)
14698{
14699 IEMEXEC_ASSERT_INSTR_LEN_RETURN(cbInstr, 3);
14700
14701 iemInitExec(pVCpu, false /*fBypassHandlers*/);
14702 VBOXSTRICTRC rcStrict = IEM_CIMPL_CALL_1(iemCImpl_lmsw, uValue);
14703 return iemUninitExecAndFiddleStatusAndMaybeReenter(pVCpu, rcStrict);
14704}
14705
14706
14707/**
14708 * Interface for HM and EM to emulate the XSETBV instruction (loads XCRx).
14709 *
14710 * Takes input values in ecx and edx:eax of the CPU context of the calling EMT.
14711 *
14712 * @returns Strict VBox status code.
14713 * @param pVCpu The cross context virtual CPU structure of the calling EMT.
14714 * @param cbInstr The instruction length in bytes.
14715 * @remarks In ring-0 not all of the state needs to be synced in.
14716 * @thread EMT(pVCpu)
14717 */
14718VMM_INT_DECL(VBOXSTRICTRC) IEMExecDecodedXsetbv(PVMCPU pVCpu, uint8_t cbInstr)
14719{
14720 IEMEXEC_ASSERT_INSTR_LEN_RETURN(cbInstr, 3);
14721
14722 iemInitExec(pVCpu, false /*fBypassHandlers*/);
14723 VBOXSTRICTRC rcStrict = IEM_CIMPL_CALL_0(iemCImpl_xsetbv);
14724 return iemUninitExecAndFiddleStatusAndMaybeReenter(pVCpu, rcStrict);
14725}
14726
14727#ifdef IN_RING3
14728
14729/**
14730 * Handles the unlikely and probably fatal merge cases.
14731 *
14732 * @returns Merged status code.
14733 * @param rcStrict Current EM status code.
14734 * @param rcStrictCommit The IOM I/O or MMIO write commit status to merge
14735 * with @a rcStrict.
14736 * @param iMemMap The memory mapping index. For error reporting only.
14737 * @param pVCpu The cross context virtual CPU structure of the calling
14738 * thread, for error reporting only.
14739 */
14740DECL_NO_INLINE(static, VBOXSTRICTRC) iemR3MergeStatusSlow(VBOXSTRICTRC rcStrict, VBOXSTRICTRC rcStrictCommit,
14741 unsigned iMemMap, PVMCPU pVCpu)
14742{
14743 if (RT_FAILURE_NP(rcStrict))
14744 return rcStrict;
14745
14746 if (RT_FAILURE_NP(rcStrictCommit))
14747 return rcStrictCommit;
14748
14749 if (rcStrict == rcStrictCommit)
14750 return rcStrictCommit;
14751
14752 AssertLogRelMsgFailed(("rcStrictCommit=%Rrc rcStrict=%Rrc iMemMap=%u fAccess=%#x FirstPg=%RGp LB %u SecondPg=%RGp LB %u\n",
14753 VBOXSTRICTRC_VAL(rcStrictCommit), VBOXSTRICTRC_VAL(rcStrict), iMemMap,
14754 pVCpu->iem.s.aMemMappings[iMemMap].fAccess,
14755 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysFirst, pVCpu->iem.s.aMemBbMappings[iMemMap].cbFirst,
14756 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysSecond, pVCpu->iem.s.aMemBbMappings[iMemMap].cbSecond));
14757 return VERR_IOM_FF_STATUS_IPE;
14758}
14759
14760
14761/**
14762 * Helper for IOMR3ProcessForceFlag.
14763 *
14764 * @returns Merged status code.
14765 * @param rcStrict Current EM status code.
14766 * @param rcStrictCommit The IOM I/O or MMIO write commit status to merge
14767 * with @a rcStrict.
14768 * @param iMemMap The memory mapping index. For error reporting only.
14769 * @param pVCpu The cross context virtual CPU structure of the calling
14770 * thread, for error reporting only.
14771 */
14772DECLINLINE(VBOXSTRICTRC) iemR3MergeStatus(VBOXSTRICTRC rcStrict, VBOXSTRICTRC rcStrictCommit, unsigned iMemMap, PVMCPU pVCpu)
14773{
14774 /* Simple. */
14775 if (RT_LIKELY(rcStrict == VINF_SUCCESS || rcStrict == VINF_EM_RAW_TO_R3))
14776 return rcStrictCommit;
14777
14778 if (RT_LIKELY(rcStrictCommit == VINF_SUCCESS))
14779 return rcStrict;
14780
14781 /* EM scheduling status codes. */
14782 if (RT_LIKELY( rcStrict >= VINF_EM_FIRST
14783 && rcStrict <= VINF_EM_LAST))
14784 {
14785 if (RT_LIKELY( rcStrictCommit >= VINF_EM_FIRST
14786 && rcStrictCommit <= VINF_EM_LAST))
14787 return rcStrict < rcStrictCommit ? rcStrict : rcStrictCommit;
14788 }
14789
14790 /* Unlikely */
14791 return iemR3MergeStatusSlow(rcStrict, rcStrictCommit, iMemMap, pVCpu);
14792}
14793
14794
14795/**
14796 * Called by force-flag handling code when VMCPU_FF_IEM is set.
14797 *
14798 * @returns Merge between @a rcStrict and what the commit operation returned.
14799 * @param pVM The cross context VM structure.
14800 * @param pVCpu The cross context virtual CPU structure of the calling EMT.
14801 * @param rcStrict The status code returned by ring-0 or raw-mode.
14802 */
14803VMMR3_INT_DECL(VBOXSTRICTRC) IEMR3ProcessForceFlag(PVM pVM, PVMCPU pVCpu, VBOXSTRICTRC rcStrict)
14804{
14805 /*
14806 * Reset the pending commit.
14807 */
14808 AssertMsg( (pVCpu->iem.s.aMemMappings[0].fAccess | pVCpu->iem.s.aMemMappings[1].fAccess | pVCpu->iem.s.aMemMappings[2].fAccess)
14809 & (IEM_ACCESS_PENDING_R3_WRITE_1ST | IEM_ACCESS_PENDING_R3_WRITE_2ND),
14810 ("%#x %#x %#x\n",
14811 pVCpu->iem.s.aMemMappings[0].fAccess, pVCpu->iem.s.aMemMappings[1].fAccess, pVCpu->iem.s.aMemMappings[2].fAccess));
14812 VMCPU_FF_CLEAR(pVCpu, VMCPU_FF_IEM);
14813
14814 /*
14815 * Commit the pending bounce buffers (usually just one).
14816 */
14817 unsigned cBufs = 0;
14818 unsigned iMemMap = RT_ELEMENTS(pVCpu->iem.s.aMemMappings);
14819 while (iMemMap-- > 0)
14820 if (pVCpu->iem.s.aMemMappings[iMemMap].fAccess & (IEM_ACCESS_PENDING_R3_WRITE_1ST | IEM_ACCESS_PENDING_R3_WRITE_2ND))
14821 {
14822 Assert(pVCpu->iem.s.aMemMappings[iMemMap].fAccess & IEM_ACCESS_TYPE_WRITE);
14823 Assert(pVCpu->iem.s.aMemMappings[iMemMap].fAccess & IEM_ACCESS_BOUNCE_BUFFERED);
14824 Assert(!pVCpu->iem.s.aMemBbMappings[iMemMap].fUnassigned);
14825
14826 uint16_t const cbFirst = pVCpu->iem.s.aMemBbMappings[iMemMap].cbFirst;
14827 uint16_t const cbSecond = pVCpu->iem.s.aMemBbMappings[iMemMap].cbSecond;
14828 uint8_t const *pbBuf = &pVCpu->iem.s.aBounceBuffers[iMemMap].ab[0];
14829
14830 if (pVCpu->iem.s.aMemMappings[iMemMap].fAccess & IEM_ACCESS_PENDING_R3_WRITE_1ST)
14831 {
14832 VBOXSTRICTRC rcStrictCommit1 = PGMPhysWrite(pVM,
14833 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysFirst,
14834 pbBuf,
14835 cbFirst,
14836 PGMACCESSORIGIN_IEM);
14837 rcStrict = iemR3MergeStatus(rcStrict, rcStrictCommit1, iMemMap, pVCpu);
14838 Log(("IEMR3ProcessForceFlag: iMemMap=%u GCPhysFirst=%RGp LB %#x %Rrc => %Rrc\n",
14839 iMemMap, pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysFirst, cbFirst,
14840 VBOXSTRICTRC_VAL(rcStrictCommit1), VBOXSTRICTRC_VAL(rcStrict)));
14841 }
14842
14843 if (pVCpu->iem.s.aMemMappings[iMemMap].fAccess & IEM_ACCESS_PENDING_R3_WRITE_2ND)
14844 {
14845 VBOXSTRICTRC rcStrictCommit2 = PGMPhysWrite(pVM,
14846 pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysSecond,
14847 pbBuf + cbFirst,
14848 cbSecond,
14849 PGMACCESSORIGIN_IEM);
14850 rcStrict = iemR3MergeStatus(rcStrict, rcStrictCommit2, iMemMap, pVCpu);
14851 Log(("IEMR3ProcessForceFlag: iMemMap=%u GCPhysSecond=%RGp LB %#x %Rrc => %Rrc\n",
14852 iMemMap, pVCpu->iem.s.aMemBbMappings[iMemMap].GCPhysSecond, cbSecond,
14853 VBOXSTRICTRC_VAL(rcStrictCommit2), VBOXSTRICTRC_VAL(rcStrict)));
14854 }
14855 cBufs++;
14856 pVCpu->iem.s.aMemMappings[iMemMap].fAccess = IEM_ACCESS_INVALID;
14857 }
14858
14859 AssertMsg(cBufs > 0 && cBufs == pVCpu->iem.s.cActiveMappings,
14860 ("cBufs=%u cActiveMappings=%u - %#x %#x %#x\n", cBufs, pVCpu->iem.s.cActiveMappings,
14861 pVCpu->iem.s.aMemMappings[0].fAccess, pVCpu->iem.s.aMemMappings[1].fAccess, pVCpu->iem.s.aMemMappings[2].fAccess));
14862 pVCpu->iem.s.cActiveMappings = 0;
14863 return rcStrict;
14864}
14865
14866#endif /* IN_RING3 */
14867
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